8066669: dust.js performance regression caused by primitive field conversion
Reviewed-by: attila, sundar
This commit is contained in:
parent
7d75c8da1a
commit
c2cd1906de
@ -465,10 +465,10 @@ final class CodeGenerator extends NodeOperatorVisitor<CodeGeneratorLexicalContex
|
|||||||
// If this is either __FILE__, __DIR__, or __LINE__ then load the property initially as Object as we'd convert
|
// If this is either __FILE__, __DIR__, or __LINE__ then load the property initially as Object as we'd convert
|
||||||
// it anyway for replaceLocationPropertyPlaceholder.
|
// it anyway for replaceLocationPropertyPlaceholder.
|
||||||
if(identNode.isCompileTimePropertyName()) {
|
if(identNode.isCompileTimePropertyName()) {
|
||||||
method.dynamicGet(Type.OBJECT, identNode.getSymbol().getName(), flags, identNode.isFunction());
|
method.dynamicGet(Type.OBJECT, identNode.getSymbol().getName(), flags, identNode.isFunction(), false);
|
||||||
replaceCompileTimeProperty();
|
replaceCompileTimeProperty();
|
||||||
} else {
|
} else {
|
||||||
dynamicGet(identNode.getSymbol().getName(), flags, identNode.isFunction());
|
dynamicGet(identNode.getSymbol().getName(), flags, identNode.isFunction(), false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -486,7 +486,7 @@ final class CodeGenerator extends NodeOperatorVisitor<CodeGeneratorLexicalContex
|
|||||||
|
|
||||||
private MethodEmitter storeFastScopeVar(final Symbol symbol, final int flags) {
|
private MethodEmitter storeFastScopeVar(final Symbol symbol, final int flags) {
|
||||||
loadFastScopeProto(symbol, true);
|
loadFastScopeProto(symbol, true);
|
||||||
method.dynamicSet(symbol.getName(), flags | CALLSITE_FAST_SCOPE);
|
method.dynamicSet(symbol.getName(), flags | CALLSITE_FAST_SCOPE, false);
|
||||||
return method;
|
return method;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -745,7 +745,7 @@ final class CodeGenerator extends NodeOperatorVisitor<CodeGeneratorLexicalContex
|
|||||||
@Override
|
@Override
|
||||||
void consumeStack() {
|
void consumeStack() {
|
||||||
final int flags = getCallSiteFlags();
|
final int flags = getCallSiteFlags();
|
||||||
dynamicGet(accessNode.getProperty(), flags, accessNode.isFunction());
|
dynamicGet(accessNode.getProperty(), flags, accessNode.isFunction(), accessNode.isIndex());
|
||||||
}
|
}
|
||||||
}.emit(baseAlreadyOnStack ? 1 : 0);
|
}.emit(baseAlreadyOnStack ? 1 : 0);
|
||||||
return false;
|
return false;
|
||||||
@ -1449,7 +1449,7 @@ final class CodeGenerator extends NodeOperatorVisitor<CodeGeneratorLexicalContex
|
|||||||
// NOTE: not using a nested OptimisticOperation on this dynamicGet, as we expect to get back
|
// NOTE: not using a nested OptimisticOperation on this dynamicGet, as we expect to get back
|
||||||
// a callable object. Nobody in their right mind would optimistically type this call site.
|
// a callable object. Nobody in their right mind would optimistically type this call site.
|
||||||
assert !node.isOptimistic();
|
assert !node.isOptimistic();
|
||||||
method.dynamicGet(node.getType(), node.getProperty(), flags, true);
|
method.dynamicGet(node.getType(), node.getProperty(), flags, true, node.isIndex());
|
||||||
method.swap();
|
method.swap();
|
||||||
argCount = loadArgs(args);
|
argCount = loadArgs(args);
|
||||||
}
|
}
|
||||||
@ -3165,7 +3165,7 @@ final class CodeGenerator extends NodeOperatorVisitor<CodeGeneratorLexicalContex
|
|||||||
if (isFastScope(identSymbol)) {
|
if (isFastScope(identSymbol)) {
|
||||||
storeFastScopeVar(identSymbol, flags);
|
storeFastScopeVar(identSymbol, flags);
|
||||||
} else {
|
} else {
|
||||||
method.dynamicSet(identNode.getName(), flags);
|
method.dynamicSet(identNode.getName(), flags, false);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
final Type identType = identNode.getType();
|
final Type identType = identNode.getType();
|
||||||
@ -4269,7 +4269,7 @@ final class CodeGenerator extends NodeOperatorVisitor<CodeGeneratorLexicalContex
|
|||||||
if (isFastScope(symbol)) {
|
if (isFastScope(symbol)) {
|
||||||
storeFastScopeVar(symbol, flags);
|
storeFastScopeVar(symbol, flags);
|
||||||
} else {
|
} else {
|
||||||
method.dynamicSet(node.getName(), flags);
|
method.dynamicSet(node.getName(), flags, false);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
final Type storeType = assignNode.getType();
|
final Type storeType = assignNode.getType();
|
||||||
@ -4286,7 +4286,7 @@ final class CodeGenerator extends NodeOperatorVisitor<CodeGeneratorLexicalContex
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean enterAccessNode(final AccessNode node) {
|
public boolean enterAccessNode(final AccessNode node) {
|
||||||
method.dynamicSet(node.getProperty(), getCallSiteFlags());
|
method.dynamicSet(node.getProperty(), getCallSiteFlags(), node.isIndex());
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -4624,11 +4624,11 @@ final class CodeGenerator extends NodeOperatorVisitor<CodeGeneratorLexicalContex
|
|||||||
* @param isMethod whether we're preferrably retrieving a function
|
* @param isMethod whether we're preferrably retrieving a function
|
||||||
* @return the current method emitter
|
* @return the current method emitter
|
||||||
*/
|
*/
|
||||||
MethodEmitter dynamicGet(final String name, final int flags, final boolean isMethod) {
|
MethodEmitter dynamicGet(final String name, final int flags, final boolean isMethod, final boolean isIndex) {
|
||||||
if(isOptimistic) {
|
if(isOptimistic) {
|
||||||
return method.dynamicGet(getOptimisticCoercedType(), name, getOptimisticFlags(flags), isMethod);
|
return method.dynamicGet(getOptimisticCoercedType(), name, getOptimisticFlags(flags), isMethod, isIndex);
|
||||||
}
|
}
|
||||||
return method.dynamicGet(resultBounds.within(expression.getType()), name, nonOptimisticFlags(flags), isMethod);
|
return method.dynamicGet(resultBounds.within(expression.getType()), name, nonOptimisticFlags(flags), isMethod, isIndex);
|
||||||
}
|
}
|
||||||
|
|
||||||
MethodEmitter dynamicGetIndex(final int flags, final boolean isMethod) {
|
MethodEmitter dynamicGetIndex(final int flags, final boolean isMethod) {
|
||||||
|
@ -34,6 +34,8 @@ import java.util.Arrays;
|
|||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.ListIterator;
|
import java.util.ListIterator;
|
||||||
|
import java.util.regex.Pattern;
|
||||||
|
import jdk.nashorn.internal.ir.AccessNode;
|
||||||
import jdk.nashorn.internal.ir.BaseNode;
|
import jdk.nashorn.internal.ir.BaseNode;
|
||||||
import jdk.nashorn.internal.ir.BinaryNode;
|
import jdk.nashorn.internal.ir.BinaryNode;
|
||||||
import jdk.nashorn.internal.ir.Block;
|
import jdk.nashorn.internal.ir.Block;
|
||||||
@ -52,6 +54,7 @@ import jdk.nashorn.internal.ir.FunctionNode;
|
|||||||
import jdk.nashorn.internal.ir.FunctionNode.CompilationState;
|
import jdk.nashorn.internal.ir.FunctionNode.CompilationState;
|
||||||
import jdk.nashorn.internal.ir.IdentNode;
|
import jdk.nashorn.internal.ir.IdentNode;
|
||||||
import jdk.nashorn.internal.ir.IfNode;
|
import jdk.nashorn.internal.ir.IfNode;
|
||||||
|
import jdk.nashorn.internal.ir.IndexNode;
|
||||||
import jdk.nashorn.internal.ir.JumpStatement;
|
import jdk.nashorn.internal.ir.JumpStatement;
|
||||||
import jdk.nashorn.internal.ir.LabelNode;
|
import jdk.nashorn.internal.ir.LabelNode;
|
||||||
import jdk.nashorn.internal.ir.LexicalContext;
|
import jdk.nashorn.internal.ir.LexicalContext;
|
||||||
@ -93,6 +96,10 @@ final class Lower extends NodeOperatorVisitor<BlockLexicalContext> implements Lo
|
|||||||
|
|
||||||
private final DebugLogger log;
|
private final DebugLogger log;
|
||||||
|
|
||||||
|
// Conservative pattern to test if element names consist of characters valid for identifiers.
|
||||||
|
// This matches any non-zero length alphanumeric string including _ and $ and not starting with a digit.
|
||||||
|
private static Pattern SAFE_PROPERTY_NAME = Pattern.compile("[a-zA-Z_$][\\w$]*");
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Constructor.
|
* Constructor.
|
||||||
*/
|
*/
|
||||||
@ -140,7 +147,7 @@ final class Lower extends NodeOperatorVisitor<BlockLexicalContext> implements Lo
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
this.log = initLogger(compiler.getContext());
|
this.log = initLogger(compiler.getContext());
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@ -180,6 +187,28 @@ final class Lower extends NodeOperatorVisitor<BlockLexicalContext> implements Lo
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Node leaveIndexNode(final IndexNode indexNode) {
|
||||||
|
final String name = getConstantPropertyName(indexNode.getIndex());
|
||||||
|
if (name != null) {
|
||||||
|
// If index node is a constant property name convert index node to access node.
|
||||||
|
assert Token.descType(indexNode.getToken()) == TokenType.LBRACKET;
|
||||||
|
return new AccessNode(indexNode.getToken(), indexNode.getFinish(), indexNode.getBase(), name);
|
||||||
|
}
|
||||||
|
return super.leaveIndexNode(indexNode);
|
||||||
|
}
|
||||||
|
|
||||||
|
// If expression is a primitive literal that is not an array index and does return its string value. Else return null.
|
||||||
|
private static String getConstantPropertyName(final Expression expression) {
|
||||||
|
if (expression instanceof LiteralNode.PrimitiveLiteralNode) {
|
||||||
|
final Object value = ((LiteralNode) expression).getValue();
|
||||||
|
if (value instanceof String && SAFE_PROPERTY_NAME.matcher((String) value).matches()) {
|
||||||
|
return (String) value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Node leaveExpressionStatement(final ExpressionStatement expressionStatement) {
|
public Node leaveExpressionStatement(final ExpressionStatement expressionStatement) {
|
||||||
final Expression expr = expressionStatement.getExpression();
|
final Expression expr = expressionStatement.getExpression();
|
||||||
|
@ -2213,10 +2213,10 @@ public class MethodEmitter implements Emitter {
|
|||||||
* @param name name of property
|
* @param name name of property
|
||||||
* @param flags call site flags
|
* @param flags call site flags
|
||||||
* @param isMethod should it prefer retrieving methods
|
* @param isMethod should it prefer retrieving methods
|
||||||
*
|
* @param isIndex is this an index operation?
|
||||||
* @return the method emitter
|
* @return the method emitter
|
||||||
*/
|
*/
|
||||||
MethodEmitter dynamicGet(final Type valueType, final String name, final int flags, final boolean isMethod) {
|
MethodEmitter dynamicGet(final Type valueType, final String name, final int flags, final boolean isMethod, final boolean isIndex) {
|
||||||
if (name.length() > LARGE_STRING_THRESHOLD) { // use getIndex for extremely long names
|
if (name.length() > LARGE_STRING_THRESHOLD) { // use getIndex for extremely long names
|
||||||
return load(name).dynamicGetIndex(valueType, flags, isMethod);
|
return load(name).dynamicGetIndex(valueType, flags, isMethod);
|
||||||
}
|
}
|
||||||
@ -2229,8 +2229,8 @@ public class MethodEmitter implements Emitter {
|
|||||||
}
|
}
|
||||||
|
|
||||||
popType(Type.SCOPE);
|
popType(Type.SCOPE);
|
||||||
method.visitInvokeDynamicInsn((isMethod ? "dyn:getMethod|getProp|getElem:" : "dyn:getProp|getElem|getMethod:") +
|
method.visitInvokeDynamicInsn(dynGetOperation(isMethod, isIndex) + ':' + NameCodec.encode(name),
|
||||||
NameCodec.encode(name), Type.getMethodDescriptor(type, Type.OBJECT), LINKERBOOTSTRAP, flags);
|
Type.getMethodDescriptor(type, Type.OBJECT), LINKERBOOTSTRAP, flags);
|
||||||
|
|
||||||
pushType(type);
|
pushType(type);
|
||||||
convert(valueType); //most probably a nop
|
convert(valueType); //most probably a nop
|
||||||
@ -2243,8 +2243,9 @@ public class MethodEmitter implements Emitter {
|
|||||||
*
|
*
|
||||||
* @param name name of property
|
* @param name name of property
|
||||||
* @param flags call site flags
|
* @param flags call site flags
|
||||||
|
* @param isIndex is this an index operation?
|
||||||
*/
|
*/
|
||||||
void dynamicSet(final String name, final int flags) {
|
void dynamicSet(final String name, final int flags, final boolean isIndex) {
|
||||||
if (name.length() > LARGE_STRING_THRESHOLD) { // use setIndex for extremely long names
|
if (name.length() > LARGE_STRING_THRESHOLD) { // use setIndex for extremely long names
|
||||||
load(name).swap().dynamicSetIndex(flags);
|
load(name).swap().dynamicSetIndex(flags);
|
||||||
return;
|
return;
|
||||||
@ -2261,7 +2262,8 @@ public class MethodEmitter implements Emitter {
|
|||||||
popType(type);
|
popType(type);
|
||||||
popType(Type.SCOPE);
|
popType(Type.SCOPE);
|
||||||
|
|
||||||
method.visitInvokeDynamicInsn("dyn:setProp|setElem:" + NameCodec.encode(name), methodDescriptor(void.class, Object.class, type.getTypeClass()), LINKERBOOTSTRAP, flags);
|
method.visitInvokeDynamicInsn(dynSetOperation(isIndex) + ':' + NameCodec.encode(name),
|
||||||
|
methodDescriptor(void.class, Object.class, type.getTypeClass()), LINKERBOOTSTRAP, flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -2294,7 +2296,7 @@ public class MethodEmitter implements Emitter {
|
|||||||
|
|
||||||
final String signature = Type.getMethodDescriptor(resultType, Type.OBJECT /*e.g STRING->OBJECT*/, index);
|
final String signature = Type.getMethodDescriptor(resultType, Type.OBJECT /*e.g STRING->OBJECT*/, index);
|
||||||
|
|
||||||
method.visitInvokeDynamicInsn(isMethod ? "dyn:getMethod|getElem|getProp" : "dyn:getElem|getProp|getMethod", signature, LINKERBOOTSTRAP, flags);
|
method.visitInvokeDynamicInsn(dynGetOperation(isMethod, true), signature, LINKERBOOTSTRAP, flags);
|
||||||
pushType(resultType);
|
pushType(resultType);
|
||||||
|
|
||||||
if (result.isBoolean()) {
|
if (result.isBoolean()) {
|
||||||
@ -2508,6 +2510,18 @@ public class MethodEmitter implements Emitter {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static String dynGetOperation(final boolean isMethod, final boolean isIndex) {
|
||||||
|
if (isMethod) {
|
||||||
|
return isIndex ? "dyn:getMethod|getElem|getProp" : "dyn:getMethod|getProp|getElem";
|
||||||
|
} else {
|
||||||
|
return isIndex ? "dyn:getElem|getProp|getMethod" : "dyn:getProp|getElem|getMethod";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static String dynSetOperation(final boolean isIndex) {
|
||||||
|
return isIndex ? "dyn:setElem|setProp" : "dyn:setProp|setElem";
|
||||||
|
}
|
||||||
|
|
||||||
private Type emitLocalVariableConversion(final LocalVariableConversion conversion, final boolean onlySymbolLiveValue) {
|
private Type emitLocalVariableConversion(final LocalVariableConversion conversion, final boolean onlySymbolLiveValue) {
|
||||||
final Type from = conversion.getFrom();
|
final Type from = conversion.getFrom();
|
||||||
final Type to = conversion.getTo();
|
final Type to = conversion.getTo();
|
||||||
|
@ -156,7 +156,7 @@ class SharedScopeCall {
|
|||||||
assert !isCall || valueType.isObject(); // Callables are always objects
|
assert !isCall || valueType.isObject(); // Callables are always objects
|
||||||
// If flags are optimistic, but we're doing a call, remove optimistic flags from the getter, as they obviously
|
// If flags are optimistic, but we're doing a call, remove optimistic flags from the getter, as they obviously
|
||||||
// only apply to the call.
|
// only apply to the call.
|
||||||
method.dynamicGet(valueType, symbol.getName(), isCall ? CodeGenerator.nonOptimisticFlags(flags) : flags, isCall);
|
method.dynamicGet(valueType, symbol.getName(), isCall ? CodeGenerator.nonOptimisticFlags(flags) : flags, isCall, false);
|
||||||
|
|
||||||
// If this is a get we're done, otherwise call the value as function.
|
// If this is a get we're done, otherwise call the value as function.
|
||||||
if (isCall) {
|
if (isCall) {
|
||||||
|
@ -28,6 +28,8 @@ package jdk.nashorn.internal.ir;
|
|||||||
import jdk.nashorn.internal.codegen.types.Type;
|
import jdk.nashorn.internal.codegen.types.Type;
|
||||||
import jdk.nashorn.internal.ir.annotations.Immutable;
|
import jdk.nashorn.internal.ir.annotations.Immutable;
|
||||||
import jdk.nashorn.internal.ir.visitor.NodeVisitor;
|
import jdk.nashorn.internal.ir.visitor.NodeVisitor;
|
||||||
|
import jdk.nashorn.internal.parser.Token;
|
||||||
|
import jdk.nashorn.internal.parser.TokenType;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* IR representation of a property access (period operator.)
|
* IR representation of a property access (period operator.)
|
||||||
@ -101,6 +103,14 @@ public final class AccessNode extends BaseNode {
|
|||||||
return property;
|
return property;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return true if this node represents an index operation normally represented as {@link IndexNode}.
|
||||||
|
* @return true if an index access.
|
||||||
|
*/
|
||||||
|
public boolean isIndex() {
|
||||||
|
return Token.descType(getToken()) == TokenType.LBRACKET;
|
||||||
|
}
|
||||||
|
|
||||||
private AccessNode setBase(final Expression base) {
|
private AccessNode setBase(final Expression base) {
|
||||||
if (this.base == base) {
|
if (this.base == base) {
|
||||||
return this;
|
return this;
|
||||||
|
@ -2001,12 +2001,11 @@ public abstract class ScriptObject implements PropertyAccess, Cloneable {
|
|||||||
|
|
||||||
if (find == null) {
|
if (find == null) {
|
||||||
switch (operator) {
|
switch (operator) {
|
||||||
|
case "getElem": // getElem only gets here if element name is constant, so treat it like a property access
|
||||||
case "getProp":
|
case "getProp":
|
||||||
return noSuchProperty(desc, request);
|
return noSuchProperty(desc, request);
|
||||||
case "getMethod":
|
case "getMethod":
|
||||||
return noSuchMethod(desc, request);
|
return noSuchMethod(desc, request);
|
||||||
case "getElem":
|
|
||||||
return createEmptyGetter(desc, explicitInstanceOfCheck, name);
|
|
||||||
default:
|
default:
|
||||||
throw new AssertionError(operator); // never invoked with any other operation
|
throw new AssertionError(operator); // never invoked with any other operation
|
||||||
}
|
}
|
||||||
|
58
nashorn/test/script/basic/JDK-8066669.js
Normal file
58
nashorn/test/script/basic/JDK-8066669.js
Normal file
@ -0,0 +1,58 @@
|
|||||||
|
/*
|
||||||
|
* 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-8066669: dust.js performance regression caused by primitive field conversion
|
||||||
|
*
|
||||||
|
* @test
|
||||||
|
* @run
|
||||||
|
*/
|
||||||
|
|
||||||
|
// Make sure index access on Java objects is working as expected.
|
||||||
|
var map = new java.util.HashMap();
|
||||||
|
|
||||||
|
map["foo"] = "bar";
|
||||||
|
map[1] = 2;
|
||||||
|
map[false] = true;
|
||||||
|
map[null] = 0;
|
||||||
|
|
||||||
|
print(map);
|
||||||
|
|
||||||
|
var keys = map.keySet().iterator();
|
||||||
|
|
||||||
|
while(keys.hasNext()) {
|
||||||
|
var key = keys.next();
|
||||||
|
print(typeof key, key);
|
||||||
|
}
|
||||||
|
|
||||||
|
print(typeof map["foo"], map["foo"]);
|
||||||
|
print(typeof map[1], map[1]);
|
||||||
|
print(typeof map[false], map[false]);
|
||||||
|
print(typeof map[null], map[null]);
|
||||||
|
|
||||||
|
print(map.foo);
|
||||||
|
print(map.false);
|
||||||
|
print(map.null);
|
||||||
|
|
||||||
|
map.foo = "baz";
|
||||||
|
print(map);
|
13
nashorn/test/script/basic/JDK-8066669.js.EXPECTED
Normal file
13
nashorn/test/script/basic/JDK-8066669.js.EXPECTED
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
{null=0, 1=2, false=true, foo=bar}
|
||||||
|
object null
|
||||||
|
number 1
|
||||||
|
boolean false
|
||||||
|
string foo
|
||||||
|
string bar
|
||||||
|
number 2
|
||||||
|
boolean true
|
||||||
|
number 0
|
||||||
|
bar
|
||||||
|
null
|
||||||
|
null
|
||||||
|
{null=0, 1=2, false=true, foo=baz}
|
@ -10,7 +10,7 @@ bar
|
|||||||
l[0]=foo
|
l[0]=foo
|
||||||
l[1]=a
|
l[1]=a
|
||||||
l[0.9]=null
|
l[0.9]=null
|
||||||
l['blah']=null
|
l['blah']=undefined
|
||||||
l[size_name]()=2
|
l[size_name]()=2
|
||||||
java.lang.IndexOutOfBoundsException: Index: 2, Size: 2
|
java.lang.IndexOutOfBoundsException: Index: 2, Size: 2
|
||||||
java.lang.IndexOutOfBoundsException: Index: Infinity, Size: 2
|
java.lang.IndexOutOfBoundsException: Index: Infinity, Size: 2
|
||||||
|
Loading…
x
Reference in New Issue
Block a user