nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/ScriptRuntime.java
author attila
Thu, 05 Mar 2015 15:43:43 +0100
changeset 29283 fb47e4d25a9f
parent 29282 a8523237b66c
child 31194 f00cb8259826
permissions -rw-r--r--
8035712: Restore some of the RuntimeCallSite specializations Reviewed-by: hannesw, lagergren
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
     1
/*
16151
97c1e756ae1e 8005663: Update copyright year to 2013
jlaskey
parents: 16147
diff changeset
     2
 * Copyright (c) 2010, 2013, Oracle and/or its affiliates. All rights reserved.
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
     4
 *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
     7
 * published by the Free Software Foundation.  Oracle designates this
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
     8
 * particular file as subject to the "Classpath" exception as provided
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
     9
 * by Oracle in the LICENSE file that accompanied this code.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    10
 *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    11
 * This code is distributed in the hope that it will be useful, but WITHOUT
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    12
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    13
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    14
 * version 2 for more details (a copy is included in the LICENSE file that
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    15
 * accompanied this code).
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    16
 *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    17
 * You should have received a copy of the GNU General Public License version
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    18
 * 2 along with this work; if not, write to the Free Software Foundation,
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    19
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    20
 *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    21
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    22
 * or visit www.oracle.com if you need additional information or have any
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    23
 * questions.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    24
 */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    25
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    26
package jdk.nashorn.internal.runtime;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    27
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    28
import static jdk.nashorn.internal.codegen.CompilerConstants.staticCall;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    29
import static jdk.nashorn.internal.codegen.CompilerConstants.staticCallNoLookup;
25250
b5a4e0ac31d1 8047359: large string size RangeError should be thrown rather than reporting negative length
sundar
parents: 25247
diff changeset
    30
import static jdk.nashorn.internal.runtime.ECMAErrors.rangeError;
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    31
import static jdk.nashorn.internal.runtime.ECMAErrors.referenceError;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    32
import static jdk.nashorn.internal.runtime.ECMAErrors.syntaxError;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    33
import static jdk.nashorn.internal.runtime.ECMAErrors.typeError;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    34
import static jdk.nashorn.internal.runtime.JSType.isRepresentableAsInt;
29282
a8523237b66c 8074031: Canonicalize is-a-JS-string tests
attila
parents: 29281
diff changeset
    35
import static jdk.nashorn.internal.runtime.JSType.isString;
29133
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
    36
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    37
import java.lang.invoke.MethodHandle;
19460
1b6d8e7b1cdf 8022782: publicLookup access failures in ScriptObject, ScriptFunction and ScriptFunction
sundar
parents: 19097
diff changeset
    38
import java.lang.invoke.MethodHandles;
26768
751b0f427090 8025435: Optimistic builtins support, implemented initial optimistic versions of push, pop, and charCodeAt
lagergren
parents: 26377
diff changeset
    39
import java.lang.invoke.SwitchPoint;
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    40
import java.lang.reflect.Array;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    41
import java.util.Collections;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    42
import java.util.Iterator;
19618
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
    43
import java.util.List;
17778
991ccffbeb13 8015459: Octane test run fails on Turkish locale
sundar
parents: 16272
diff changeset
    44
import java.util.Locale;
19618
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
    45
import java.util.Map;
16226
0e4f37e6cc40 8007915: Nashorn IR, codegen, parser packages and Context instance should be inaccessible to user code
sundar
parents: 16206
diff changeset
    46
import java.util.NoSuchElementException;
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    47
import java.util.Objects;
16234
86cb162cec6c 8008085: Integrate Dynalink source code into Nashorn codebase
attila
parents: 16232
diff changeset
    48
import jdk.internal.dynalink.beans.StaticClass;
19889
63af9358d0dc 8024615: Refactor ScriptObjectMirror and JSObject to support external JSObject implementations
sundar
parents: 19884
diff changeset
    49
import jdk.nashorn.api.scripting.JSObject;
17978
750d0582d8e2 8015830: Javascript mapping of ScriptEngine bindings does not expose keys
sundar
parents: 17778
diff changeset
    50
import jdk.nashorn.api.scripting.ScriptObjectMirror;
26768
751b0f427090 8025435: Optimistic builtins support, implemented initial optimistic versions of push, pop, and charCodeAt
lagergren
parents: 26377
diff changeset
    51
import jdk.nashorn.internal.codegen.ApplySpecialization;
24719
f726e9d67629 8035820: Optimistic recompilation
attila
parents: 21446
diff changeset
    52
import jdk.nashorn.internal.codegen.CompilerConstants;
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    53
import jdk.nashorn.internal.codegen.CompilerConstants.Call;
16226
0e4f37e6cc40 8007915: Nashorn IR, codegen, parser packages and Context instance should be inaccessible to user code
sundar
parents: 16206
diff changeset
    54
import jdk.nashorn.internal.ir.debug.JSONWriter;
21446
2fcb0f7e407f 8026955: for-in should convert primitive values to object
hannesw
parents: 19889
diff changeset
    55
import jdk.nashorn.internal.objects.Global;
25247
c0f911459863 8046013: TypeError: Cannot apply "with" to non script object
sundar
parents: 24762
diff changeset
    56
import jdk.nashorn.internal.objects.NativeObject;
16226
0e4f37e6cc40 8007915: Nashorn IR, codegen, parser packages and Context instance should be inaccessible to user code
sundar
parents: 16206
diff changeset
    57
import jdk.nashorn.internal.parser.Lexer;
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    58
import jdk.nashorn.internal.runtime.linker.Bootstrap;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    59
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    60
/**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    61
 * Utilities to be called by JavaScript runtime API and generated classes.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    62
 */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    63
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    64
public final class ScriptRuntime {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    65
    private ScriptRuntime() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    66
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    67
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    68
    /** Singleton representing the empty array object '[]' */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    69
    public static final Object[] EMPTY_ARRAY = new Object[0];
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    70
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    71
    /** Unique instance of undefined. */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    72
    public static final Undefined UNDEFINED = Undefined.getUndefined();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    73
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    74
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    75
     * Unique instance of undefined used to mark empty array slots.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    76
     * Can't escape the array.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    77
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    78
    public static final Undefined EMPTY = Undefined.getEmpty();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    79
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    80
    /** Method handle to generic + operator, operating on objects */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    81
    public static final Call ADD = staticCallNoLookup(ScriptRuntime.class, "ADD", Object.class, Object.class, Object.class);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    82
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    83
    /** Method handle to generic === operator, operating on objects */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    84
    public static final Call EQ_STRICT = staticCallNoLookup(ScriptRuntime.class, "EQ_STRICT", boolean.class, Object.class, Object.class);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    85
16240
e1468b33e201 8008239: Unpublicized parts of the code generator package that were only package internal.
lagergren
parents: 16234
diff changeset
    86
    /** Method handle used to enter a {@code with} scope at runtime. */
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    87
    public static final Call OPEN_WITH = staticCallNoLookup(ScriptRuntime.class, "openWith", ScriptObject.class, ScriptObject.class, Object.class);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    88
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    89
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    90
     * Method used to place a scope's variable into the Global scope, which has to be done for the
16240
e1468b33e201 8008239: Unpublicized parts of the code generator package that were only package internal.
lagergren
parents: 16234
diff changeset
    91
     * properties declared at outermost script level.
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    92
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    93
    public static final Call MERGE_SCOPE = staticCallNoLookup(ScriptRuntime.class, "mergeScope", ScriptObject.class, ScriptObject.class);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    94
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    95
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    96
     * Return an appropriate iterator for the elements in a for-in construct
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    97
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    98
    public static final Call TO_PROPERTY_ITERATOR = staticCallNoLookup(ScriptRuntime.class, "toPropertyIterator", Iterator.class, Object.class);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    99
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   100
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   101
     * Return an appropriate iterator for the elements in a for-each construct
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   102
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   103
    public static final Call TO_VALUE_ITERATOR = staticCallNoLookup(ScriptRuntime.class, "toValueIterator", Iterator.class, Object.class);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   104
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   105
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   106
      * Method handle for apply. Used from {@link ScriptFunction} for looking up calls to
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   107
      * call sites that are known to be megamorphic. Using an invoke dynamic here would
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   108
      * lead to the JVM deoptimizing itself to death
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   109
      */
19460
1b6d8e7b1cdf 8022782: publicLookup access failures in ScriptObject, ScriptFunction and ScriptFunction
sundar
parents: 19097
diff changeset
   110
    public static final Call APPLY = staticCall(MethodHandles.lookup(), ScriptRuntime.class, "apply", Object.class, ScriptFunction.class, Object.class, Object[].class);
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   111
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   112
    /**
26377
028dad61662f 8051889: Implement block scoping in symbol assignment and scope computation
hannesw
parents: 26068
diff changeset
   113
     * Throws a reference error for an undefined variable.
028dad61662f 8051889: Implement block scoping in symbol assignment and scope computation
hannesw
parents: 26068
diff changeset
   114
     */
028dad61662f 8051889: Implement block scoping in symbol assignment and scope computation
hannesw
parents: 26068
diff changeset
   115
    public static final Call THROW_REFERENCE_ERROR = staticCall(MethodHandles.lookup(), ScriptRuntime.class, "throwReferenceError", void.class, String.class);
028dad61662f 8051889: Implement block scoping in symbol assignment and scope computation
hannesw
parents: 26068
diff changeset
   116
028dad61662f 8051889: Implement block scoping in symbol assignment and scope computation
hannesw
parents: 26068
diff changeset
   117
    /**
29281
8cc2618a07aa 8073707: const re-assignment should not reported as a early error
hannesw
parents: 29133
diff changeset
   118
     * Throws a reference error for an undefined variable.
8cc2618a07aa 8073707: const re-assignment should not reported as a early error
hannesw
parents: 29133
diff changeset
   119
     */
8cc2618a07aa 8073707: const re-assignment should not reported as a early error
hannesw
parents: 29133
diff changeset
   120
    public static final Call THROW_CONST_TYPE_ERROR = staticCall(MethodHandles.lookup(), ScriptRuntime.class, "throwConstTypeError", void.class, String.class);
8cc2618a07aa 8073707: const re-assignment should not reported as a early error
hannesw
parents: 29133
diff changeset
   121
8cc2618a07aa 8073707: const re-assignment should not reported as a early error
hannesw
parents: 29133
diff changeset
   122
    /**
26768
751b0f427090 8025435: Optimistic builtins support, implemented initial optimistic versions of push, pop, and charCodeAt
lagergren
parents: 26377
diff changeset
   123
     * Used to invalidate builtin names, e.g "Function" mapping to all properties in Function.prototype and Function.prototype itself.
751b0f427090 8025435: Optimistic builtins support, implemented initial optimistic versions of push, pop, and charCodeAt
lagergren
parents: 26377
diff changeset
   124
     */
751b0f427090 8025435: Optimistic builtins support, implemented initial optimistic versions of push, pop, and charCodeAt
lagergren
parents: 26377
diff changeset
   125
    public static final Call INVALIDATE_RESERVED_BUILTIN_NAME = staticCallNoLookup(ScriptRuntime.class, "invalidateReservedBuiltinName", void.class, String.class);
751b0f427090 8025435: Optimistic builtins support, implemented initial optimistic versions of push, pop, and charCodeAt
lagergren
parents: 26377
diff changeset
   126
751b0f427090 8025435: Optimistic builtins support, implemented initial optimistic versions of push, pop, and charCodeAt
lagergren
parents: 26377
diff changeset
   127
    /**
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   128
     * Converts a switch tag value to a simple integer. deflt value if it can't.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   129
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   130
     * @param tag   Switch statement tag value.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   131
     * @param deflt default to use if not convertible.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   132
     * @return int tag value (or deflt.)
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   133
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   134
    public static int switchTagAsInt(final Object tag, final int deflt) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   135
        if (tag instanceof Number) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   136
            final double d = ((Number)tag).doubleValue();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   137
            if (isRepresentableAsInt(d)) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   138
                return (int)d;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   139
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   140
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   141
        return deflt;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   142
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   143
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   144
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   145
     * Converts a switch tag value to a simple integer. deflt value if it can't.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   146
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   147
     * @param tag   Switch statement tag value.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   148
     * @param deflt default to use if not convertible.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   149
     * @return int tag value (or deflt.)
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   150
     */
18628
012cd852f881 8019488: switch on literals result in NoSuchMethodError or VerifyError
sundar
parents: 18614
diff changeset
   151
    public static int switchTagAsInt(final boolean tag, final int deflt) {
012cd852f881 8019488: switch on literals result in NoSuchMethodError or VerifyError
sundar
parents: 18614
diff changeset
   152
        return deflt;
012cd852f881 8019488: switch on literals result in NoSuchMethodError or VerifyError
sundar
parents: 18614
diff changeset
   153
    }
012cd852f881 8019488: switch on literals result in NoSuchMethodError or VerifyError
sundar
parents: 18614
diff changeset
   154
012cd852f881 8019488: switch on literals result in NoSuchMethodError or VerifyError
sundar
parents: 18614
diff changeset
   155
    /**
012cd852f881 8019488: switch on literals result in NoSuchMethodError or VerifyError
sundar
parents: 18614
diff changeset
   156
     * Converts a switch tag value to a simple integer. deflt value if it can't.
012cd852f881 8019488: switch on literals result in NoSuchMethodError or VerifyError
sundar
parents: 18614
diff changeset
   157
     *
012cd852f881 8019488: switch on literals result in NoSuchMethodError or VerifyError
sundar
parents: 18614
diff changeset
   158
     * @param tag   Switch statement tag value.
012cd852f881 8019488: switch on literals result in NoSuchMethodError or VerifyError
sundar
parents: 18614
diff changeset
   159
     * @param deflt default to use if not convertible.
012cd852f881 8019488: switch on literals result in NoSuchMethodError or VerifyError
sundar
parents: 18614
diff changeset
   160
     * @return int tag value (or deflt.)
012cd852f881 8019488: switch on literals result in NoSuchMethodError or VerifyError
sundar
parents: 18614
diff changeset
   161
     */
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   162
    public static int switchTagAsInt(final long tag, final int deflt) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   163
        return isRepresentableAsInt(tag) ? (int)tag : deflt;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   164
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   165
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   166
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   167
     * Converts a switch tag value to a simple integer. deflt value if it can't.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   168
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   169
     * @param tag   Switch statement tag value.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   170
     * @param deflt default to use if not convertible.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   171
     * @return int tag value (or deflt.)
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   172
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   173
    public static int switchTagAsInt(final double tag, final int deflt) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   174
        return isRepresentableAsInt(tag) ? (int)tag : deflt;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   175
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   176
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   177
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   178
     * This is the builtin implementation of {@code Object.prototype.toString}
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   179
     * @param self reference
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   180
     * @return string representation as object
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   181
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   182
    public static String builtinObjectToString(final Object self) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   183
        String className;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   184
        // Spec tells us to convert primitives by ToObject..
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   185
        // But we don't need to -- all we need is the right class name
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   186
        // of the corresponding primitive wrapper type.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   187
24752
c835f368e8e0 8043002: Improve performance of Nashorn equality operators
attila
parents: 24745
diff changeset
   188
        final JSType type = JSType.ofNoFunction(self);
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   189
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   190
        switch (type) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   191
        case BOOLEAN:
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   192
            className = "Boolean";
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   193
            break;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   194
        case NUMBER:
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   195
            className = "Number";
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   196
            break;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   197
        case STRING:
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   198
            className = "String";
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   199
            break;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   200
        // special case of null and undefined
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   201
        case NULL:
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   202
            className = "Null";
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   203
            break;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   204
        case UNDEFINED:
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   205
            className = "Undefined";
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   206
            break;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   207
        case OBJECT:
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   208
            if (self instanceof ScriptObject) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   209
                className = ((ScriptObject)self).getClassName();
19889
63af9358d0dc 8024615: Refactor ScriptObjectMirror and JSObject to support external JSObject implementations
sundar
parents: 19884
diff changeset
   210
            } else if (self instanceof JSObject) {
63af9358d0dc 8024615: Refactor ScriptObjectMirror and JSObject to support external JSObject implementations
sundar
parents: 19884
diff changeset
   211
                className = ((JSObject)self).getClassName();
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   212
            } else {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   213
                className = self.getClass().getName();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   214
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   215
            break;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   216
        default:
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   217
            // Nashorn extension: use Java class name
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   218
            className = self.getClass().getName();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   219
            break;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   220
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   221
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   222
        final StringBuilder sb = new StringBuilder();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   223
        sb.append("[object ");
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   224
        sb.append(className);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   225
        sb.append(']');
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   226
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   227
        return sb.toString();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   228
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   229
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   230
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   231
     * This is called whenever runtime wants to throw an error and wants to provide
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   232
     * meaningful information about an object. We don't want to call toString which
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   233
     * ends up calling "toString" from script world which may itself throw error.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   234
     * When we want to throw an error, we don't additional error from script land
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   235
     * -- which may sometimes lead to infinite recursion.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   236
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   237
     * @param obj Object to converted to String safely (without calling user script)
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   238
     * @return safe String representation of the given object
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   239
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   240
    public static String safeToString(final Object obj) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   241
        return JSType.toStringImpl(obj, true);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   242
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   243
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   244
    /**
19618
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   245
     * Returns an iterator over property identifiers used in the {@code for...in} statement. Note that the ECMAScript
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   246
     * 5.1 specification, chapter 12.6.4. uses the terminology "property names", which seems to imply that the property
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   247
     * identifiers are expected to be strings, but this is not actually spelled out anywhere, and Nashorn will in some
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   248
     * cases deviate from this. Namely, we guarantee to always return an iterator over {@link String} values for any
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   249
     * built-in JavaScript object. We will however return an iterator over {@link Integer} objects for native Java
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   250
     * arrays and {@link List} objects, as well as arbitrary objects representing keys of a {@link Map}. Therefore, the
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   251
     * expression {@code typeof i} within a {@code for(i in obj)} statement can return something other than
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   252
     * {@code string} when iterating over native Java arrays, {@code List}, and {@code Map} objects.
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   253
     * @param obj object to iterate on.
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   254
     * @return iterator over the object's property names.
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   255
     */
19618
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   256
    public static Iterator<?> toPropertyIterator(final Object obj) {
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   257
        if (obj instanceof ScriptObject) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   258
            return ((ScriptObject)obj).propertyIterator();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   259
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   260
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   261
        if (obj != null && obj.getClass().isArray()) {
19618
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   262
            return new RangeIterator(Array.getLength(obj));
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   263
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   264
19889
63af9358d0dc 8024615: Refactor ScriptObjectMirror and JSObject to support external JSObject implementations
sundar
parents: 19884
diff changeset
   265
        if (obj instanceof JSObject) {
63af9358d0dc 8024615: Refactor ScriptObjectMirror and JSObject to support external JSObject implementations
sundar
parents: 19884
diff changeset
   266
            return ((JSObject)obj).keySet().iterator();
17978
750d0582d8e2 8015830: Javascript mapping of ScriptEngine bindings does not expose keys
sundar
parents: 17778
diff changeset
   267
        }
750d0582d8e2 8015830: Javascript mapping of ScriptEngine bindings does not expose keys
sundar
parents: 17778
diff changeset
   268
19618
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   269
        if (obj instanceof List) {
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   270
            return new RangeIterator(((List<?>)obj).size());
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   271
        }
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   272
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   273
        if (obj instanceof Map) {
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   274
            return ((Map<?,?>)obj).keySet().iterator();
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   275
        }
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   276
21446
2fcb0f7e407f 8026955: for-in should convert primitive values to object
hannesw
parents: 19889
diff changeset
   277
        final Object wrapped = Global.instance().wrapAsObject(obj);
2fcb0f7e407f 8026955: for-in should convert primitive values to object
hannesw
parents: 19889
diff changeset
   278
        if (wrapped instanceof ScriptObject) {
2fcb0f7e407f 8026955: for-in should convert primitive values to object
hannesw
parents: 19889
diff changeset
   279
            return ((ScriptObject)wrapped).propertyIterator();
2fcb0f7e407f 8026955: for-in should convert primitive values to object
hannesw
parents: 19889
diff changeset
   280
        }
2fcb0f7e407f 8026955: for-in should convert primitive values to object
hannesw
parents: 19889
diff changeset
   281
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   282
        return Collections.emptyIterator();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   283
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   284
19618
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   285
    private static final class RangeIterator implements Iterator<Integer> {
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   286
        private final int length;
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   287
        private int index;
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   288
24725
7bb1f687a852 8033334: Make sure that scope depth information is maintained in the RecompilableScriptFunctionDatas, to avoid unnecessary slow proto linkage when doing on demand compilation
lagergren
parents: 24719
diff changeset
   289
        RangeIterator(final int length) {
19618
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   290
            this.length = length;
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   291
        }
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   292
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   293
        @Override
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   294
        public boolean hasNext() {
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   295
            return index < length;
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   296
        }
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   297
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   298
        @Override
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   299
        public Integer next() {
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   300
            return index++;
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   301
        }
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   302
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   303
        @Override
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   304
        public void remove() {
26768
751b0f427090 8025435: Optimistic builtins support, implemented initial optimistic versions of push, pop, and charCodeAt
lagergren
parents: 26377
diff changeset
   305
            throw new UnsupportedOperationException("remove");
19618
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   306
        }
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   307
    }
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   308
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   309
    /**
19618
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   310
     * Returns an iterator over property values used in the {@code for each...in} statement. Aside from built-in JS
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   311
     * objects, it also operates on Java arrays, any {@link Iterable}, as well as on {@link Map} objects, iterating over
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   312
     * map values.
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   313
     * @param obj object to iterate on.
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   314
     * @return iterator over the object's property values.
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   315
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   316
    public static Iterator<?> toValueIterator(final Object obj) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   317
        if (obj instanceof ScriptObject) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   318
            return ((ScriptObject)obj).valueIterator();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   319
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   320
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   321
        if (obj != null && obj.getClass().isArray()) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   322
            final Object array  = obj;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   323
            final int    length = Array.getLength(obj);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   324
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   325
            return new Iterator<Object>() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   326
                private int index = 0;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   327
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   328
                @Override
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   329
                public boolean hasNext() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   330
                    return index < length;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   331
                }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   332
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   333
                @Override
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   334
                public Object next() {
16226
0e4f37e6cc40 8007915: Nashorn IR, codegen, parser packages and Context instance should be inaccessible to user code
sundar
parents: 16206
diff changeset
   335
                    if (index >= length) {
0e4f37e6cc40 8007915: Nashorn IR, codegen, parser packages and Context instance should be inaccessible to user code
sundar
parents: 16206
diff changeset
   336
                        throw new NoSuchElementException();
0e4f37e6cc40 8007915: Nashorn IR, codegen, parser packages and Context instance should be inaccessible to user code
sundar
parents: 16206
diff changeset
   337
                    }
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   338
                    return Array.get(array, index++);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   339
                }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   340
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   341
                @Override
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   342
                public void remove() {
26768
751b0f427090 8025435: Optimistic builtins support, implemented initial optimistic versions of push, pop, and charCodeAt
lagergren
parents: 26377
diff changeset
   343
                    throw new UnsupportedOperationException("remove");
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   344
                }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   345
            };
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   346
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   347
19889
63af9358d0dc 8024615: Refactor ScriptObjectMirror and JSObject to support external JSObject implementations
sundar
parents: 19884
diff changeset
   348
        if (obj instanceof JSObject) {
63af9358d0dc 8024615: Refactor ScriptObjectMirror and JSObject to support external JSObject implementations
sundar
parents: 19884
diff changeset
   349
            return ((JSObject)obj).values().iterator();
17978
750d0582d8e2 8015830: Javascript mapping of ScriptEngine bindings does not expose keys
sundar
parents: 17778
diff changeset
   350
        }
750d0582d8e2 8015830: Javascript mapping of ScriptEngine bindings does not expose keys
sundar
parents: 17778
diff changeset
   351
19618
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   352
        if (obj instanceof Map) {
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   353
            return ((Map<?,?>)obj).values().iterator();
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   354
        }
6b73157185e0 8022903: Enhance for-in and for-each for Lists and Maps
attila
parents: 19460
diff changeset
   355
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   356
        if (obj instanceof Iterable) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   357
            return ((Iterable<?>)obj).iterator();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   358
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   359
21446
2fcb0f7e407f 8026955: for-in should convert primitive values to object
hannesw
parents: 19889
diff changeset
   360
        final Object wrapped = Global.instance().wrapAsObject(obj);
2fcb0f7e407f 8026955: for-in should convert primitive values to object
hannesw
parents: 19889
diff changeset
   361
        if (wrapped instanceof ScriptObject) {
2fcb0f7e407f 8026955: for-in should convert primitive values to object
hannesw
parents: 19889
diff changeset
   362
            return ((ScriptObject)wrapped).valueIterator();
2fcb0f7e407f 8026955: for-in should convert primitive values to object
hannesw
parents: 19889
diff changeset
   363
        }
2fcb0f7e407f 8026955: for-in should convert primitive values to object
hannesw
parents: 19889
diff changeset
   364
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   365
        return Collections.emptyIterator();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   366
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   367
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   368
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   369
     * Merge a scope into its prototype's map.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   370
     * Merge a scope into its prototype.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   371
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   372
     * @param scope Scope to merge.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   373
     * @return prototype object after merge
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   374
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   375
    public static ScriptObject mergeScope(final ScriptObject scope) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   376
        final ScriptObject global = scope.getProto();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   377
        global.addBoundProperties(scope);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   378
        return global;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   379
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   380
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   381
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   382
     * Call a function given self and args. If the number of the arguments is known in advance, you can likely achieve
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   383
     * better performance by {@link Bootstrap#createDynamicInvoker(String, Class, Class...) creating a dynamic invoker}
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   384
     * for operation {@code "dyn:call"}, then using its {@link MethodHandle#invokeExact(Object...)} method instead.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   385
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   386
     * @param target ScriptFunction object.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   387
     * @param self   Receiver in call.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   388
     * @param args   Call arguments.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   389
     * @return Call result.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   390
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   391
    public static Object apply(final ScriptFunction target, final Object self, final Object... args) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   392
        try {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   393
            return target.invoke(self, args);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   394
        } catch (final RuntimeException | Error e) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   395
            throw e;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   396
        } catch (final Throwable t) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   397
            throw new RuntimeException(t);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   398
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   399
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   400
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   401
    /**
26377
028dad61662f 8051889: Implement block scoping in symbol assignment and scope computation
hannesw
parents: 26068
diff changeset
   402
     * Throws a reference error for an undefined variable.
028dad61662f 8051889: Implement block scoping in symbol assignment and scope computation
hannesw
parents: 26068
diff changeset
   403
     *
028dad61662f 8051889: Implement block scoping in symbol assignment and scope computation
hannesw
parents: 26068
diff changeset
   404
     * @param name the variable name
028dad61662f 8051889: Implement block scoping in symbol assignment and scope computation
hannesw
parents: 26068
diff changeset
   405
     */
028dad61662f 8051889: Implement block scoping in symbol assignment and scope computation
hannesw
parents: 26068
diff changeset
   406
    public static void throwReferenceError(final String name) {
028dad61662f 8051889: Implement block scoping in symbol assignment and scope computation
hannesw
parents: 26068
diff changeset
   407
        throw referenceError("not.defined", name);
028dad61662f 8051889: Implement block scoping in symbol assignment and scope computation
hannesw
parents: 26068
diff changeset
   408
    }
028dad61662f 8051889: Implement block scoping in symbol assignment and scope computation
hannesw
parents: 26068
diff changeset
   409
028dad61662f 8051889: Implement block scoping in symbol assignment and scope computation
hannesw
parents: 26068
diff changeset
   410
    /**
29281
8cc2618a07aa 8073707: const re-assignment should not reported as a early error
hannesw
parents: 29133
diff changeset
   411
     * Throws a type error for an assignment to a const.
8cc2618a07aa 8073707: const re-assignment should not reported as a early error
hannesw
parents: 29133
diff changeset
   412
     *
8cc2618a07aa 8073707: const re-assignment should not reported as a early error
hannesw
parents: 29133
diff changeset
   413
     * @param name the const name
8cc2618a07aa 8073707: const re-assignment should not reported as a early error
hannesw
parents: 29133
diff changeset
   414
     */
8cc2618a07aa 8073707: const re-assignment should not reported as a early error
hannesw
parents: 29133
diff changeset
   415
    public static void throwConstTypeError(final String name) {
8cc2618a07aa 8073707: const re-assignment should not reported as a early error
hannesw
parents: 29133
diff changeset
   416
        throw typeError("assign.constant", name);
8cc2618a07aa 8073707: const re-assignment should not reported as a early error
hannesw
parents: 29133
diff changeset
   417
    }
8cc2618a07aa 8073707: const re-assignment should not reported as a early error
hannesw
parents: 29133
diff changeset
   418
8cc2618a07aa 8073707: const re-assignment should not reported as a early error
hannesw
parents: 29133
diff changeset
   419
    /**
18614
addca7a10167 8015959: Can't call foreign constructor
sundar
parents: 18334
diff changeset
   420
     * Call a script function as a constructor with given args.
addca7a10167 8015959: Can't call foreign constructor
sundar
parents: 18334
diff changeset
   421
     *
addca7a10167 8015959: Can't call foreign constructor
sundar
parents: 18334
diff changeset
   422
     * @param target ScriptFunction object.
addca7a10167 8015959: Can't call foreign constructor
sundar
parents: 18334
diff changeset
   423
     * @param args   Call arguments.
addca7a10167 8015959: Can't call foreign constructor
sundar
parents: 18334
diff changeset
   424
     * @return Constructor call result.
addca7a10167 8015959: Can't call foreign constructor
sundar
parents: 18334
diff changeset
   425
     */
addca7a10167 8015959: Can't call foreign constructor
sundar
parents: 18334
diff changeset
   426
    public static Object construct(final ScriptFunction target, final Object... args) {
addca7a10167 8015959: Can't call foreign constructor
sundar
parents: 18334
diff changeset
   427
        try {
addca7a10167 8015959: Can't call foreign constructor
sundar
parents: 18334
diff changeset
   428
            return target.construct(args);
addca7a10167 8015959: Can't call foreign constructor
sundar
parents: 18334
diff changeset
   429
        } catch (final RuntimeException | Error e) {
addca7a10167 8015959: Can't call foreign constructor
sundar
parents: 18334
diff changeset
   430
            throw e;
addca7a10167 8015959: Can't call foreign constructor
sundar
parents: 18334
diff changeset
   431
        } catch (final Throwable t) {
addca7a10167 8015959: Can't call foreign constructor
sundar
parents: 18334
diff changeset
   432
            throw new RuntimeException(t);
addca7a10167 8015959: Can't call foreign constructor
sundar
parents: 18334
diff changeset
   433
        }
addca7a10167 8015959: Can't call foreign constructor
sundar
parents: 18334
diff changeset
   434
    }
addca7a10167 8015959: Can't call foreign constructor
sundar
parents: 18334
diff changeset
   435
addca7a10167 8015959: Can't call foreign constructor
sundar
parents: 18334
diff changeset
   436
    /**
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   437
     * Generic implementation of ECMA 9.12 - SameValue algorithm
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   438
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   439
     * @param x first value to compare
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   440
     * @param y second value to compare
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   441
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   442
     * @return true if both objects have the same value
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   443
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   444
    public static boolean sameValue(final Object x, final Object y) {
24752
c835f368e8e0 8043002: Improve performance of Nashorn equality operators
attila
parents: 24745
diff changeset
   445
        final JSType xType = JSType.ofNoFunction(x);
c835f368e8e0 8043002: Improve performance of Nashorn equality operators
attila
parents: 24745
diff changeset
   446
        final JSType yType = JSType.ofNoFunction(y);
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   447
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   448
        if (xType != yType) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   449
            return false;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   450
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   451
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   452
        if (xType == JSType.UNDEFINED || xType == JSType.NULL) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   453
            return true;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   454
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   455
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   456
        if (xType == JSType.NUMBER) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   457
            final double xVal = ((Number)x).doubleValue();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   458
            final double yVal = ((Number)y).doubleValue();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   459
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   460
            if (Double.isNaN(xVal) && Double.isNaN(yVal)) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   461
                return true;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   462
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   463
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   464
            // checking for xVal == -0.0 and yVal == +0.0 or vice versa
24725
7bb1f687a852 8033334: Make sure that scope depth information is maintained in the RecompilableScriptFunctionDatas, to avoid unnecessary slow proto linkage when doing on demand compilation
lagergren
parents: 24719
diff changeset
   465
            if (xVal == 0.0 && Double.doubleToLongBits(xVal) != Double.doubleToLongBits(yVal)) {
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   466
                return false;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   467
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   468
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   469
            return xVal == yVal;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   470
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   471
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   472
        if (xType == JSType.STRING || yType == JSType.BOOLEAN) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   473
            return x.equals(y);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   474
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   475
24725
7bb1f687a852 8033334: Make sure that scope depth information is maintained in the RecompilableScriptFunctionDatas, to avoid unnecessary slow proto linkage when doing on demand compilation
lagergren
parents: 24719
diff changeset
   476
        return x == y;
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   477
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   478
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   479
    /**
16226
0e4f37e6cc40 8007915: Nashorn IR, codegen, parser packages and Context instance should be inaccessible to user code
sundar
parents: 16206
diff changeset
   480
     * Returns AST as JSON compatible string. This is used to
0e4f37e6cc40 8007915: Nashorn IR, codegen, parser packages and Context instance should be inaccessible to user code
sundar
parents: 16206
diff changeset
   481
     * implement "parse" function in resources/parse.js script.
0e4f37e6cc40 8007915: Nashorn IR, codegen, parser packages and Context instance should be inaccessible to user code
sundar
parents: 16206
diff changeset
   482
     *
0e4f37e6cc40 8007915: Nashorn IR, codegen, parser packages and Context instance should be inaccessible to user code
sundar
parents: 16206
diff changeset
   483
     * @param code code to be parsed
0e4f37e6cc40 8007915: Nashorn IR, codegen, parser packages and Context instance should be inaccessible to user code
sundar
parents: 16206
diff changeset
   484
     * @param name name of the code source (used for location)
0e4f37e6cc40 8007915: Nashorn IR, codegen, parser packages and Context instance should be inaccessible to user code
sundar
parents: 16206
diff changeset
   485
     * @param includeLoc tells whether to include location information for nodes or not
0e4f37e6cc40 8007915: Nashorn IR, codegen, parser packages and Context instance should be inaccessible to user code
sundar
parents: 16206
diff changeset
   486
     * @return JSON string representation of AST of the supplied code
0e4f37e6cc40 8007915: Nashorn IR, codegen, parser packages and Context instance should be inaccessible to user code
sundar
parents: 16206
diff changeset
   487
     */
0e4f37e6cc40 8007915: Nashorn IR, codegen, parser packages and Context instance should be inaccessible to user code
sundar
parents: 16206
diff changeset
   488
    public static String parse(final String code, final String name, final boolean includeLoc) {
24745
3a6e1477362b 8041434: Add synchronization to the common global constants structure
lagergren
parents: 24733
diff changeset
   489
        return JSONWriter.parse(Context.getContextTrusted(), code, name, includeLoc);
16226
0e4f37e6cc40 8007915: Nashorn IR, codegen, parser packages and Context instance should be inaccessible to user code
sundar
parents: 16206
diff changeset
   490
    }
0e4f37e6cc40 8007915: Nashorn IR, codegen, parser packages and Context instance should be inaccessible to user code
sundar
parents: 16206
diff changeset
   491
0e4f37e6cc40 8007915: Nashorn IR, codegen, parser packages and Context instance should be inaccessible to user code
sundar
parents: 16206
diff changeset
   492
    /**
0e4f37e6cc40 8007915: Nashorn IR, codegen, parser packages and Context instance should be inaccessible to user code
sundar
parents: 16206
diff changeset
   493
     * Test whether a char is valid JavaScript whitespace
0e4f37e6cc40 8007915: Nashorn IR, codegen, parser packages and Context instance should be inaccessible to user code
sundar
parents: 16206
diff changeset
   494
     * @param ch a char
0e4f37e6cc40 8007915: Nashorn IR, codegen, parser packages and Context instance should be inaccessible to user code
sundar
parents: 16206
diff changeset
   495
     * @return true if valid JavaScript whitespace
0e4f37e6cc40 8007915: Nashorn IR, codegen, parser packages and Context instance should be inaccessible to user code
sundar
parents: 16206
diff changeset
   496
     */
0e4f37e6cc40 8007915: Nashorn IR, codegen, parser packages and Context instance should be inaccessible to user code
sundar
parents: 16206
diff changeset
   497
    public static boolean isJSWhitespace(final char ch) {
0e4f37e6cc40 8007915: Nashorn IR, codegen, parser packages and Context instance should be inaccessible to user code
sundar
parents: 16206
diff changeset
   498
        return Lexer.isJSWhitespace(ch);
0e4f37e6cc40 8007915: Nashorn IR, codegen, parser packages and Context instance should be inaccessible to user code
sundar
parents: 16206
diff changeset
   499
    }
0e4f37e6cc40 8007915: Nashorn IR, codegen, parser packages and Context instance should be inaccessible to user code
sundar
parents: 16206
diff changeset
   500
0e4f37e6cc40 8007915: Nashorn IR, codegen, parser packages and Context instance should be inaccessible to user code
sundar
parents: 16206
diff changeset
   501
    /**
24719
f726e9d67629 8035820: Optimistic recompilation
attila
parents: 21446
diff changeset
   502
     * Entering a {@code with} node requires new scope. This is the implementation. When exiting the with statement,
f726e9d67629 8035820: Optimistic recompilation
attila
parents: 21446
diff changeset
   503
     * use {@link ScriptObject#getProto()} on the scope.
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   504
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   505
     * @param scope      existing scope
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   506
     * @param expression expression in with
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   507
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   508
     * @return {@link WithObject} that is the new scope
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   509
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   510
    public static ScriptObject openWith(final ScriptObject scope, final Object expression) {
23375
a1110f2cbe75 8037400: Remove getInitialMap getters and GlobalObject interface
sundar
parents: 21446
diff changeset
   511
        final Global global = Context.getGlobal();
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   512
        if (expression == UNDEFINED) {
16256
f2d9a0c49914 8007002: Replace implicit exception throwing methods with explicit throws - simplify control flow and remove useless code
lagergren
parents: 16240
diff changeset
   513
            throw typeError(global, "cant.apply.with.to.undefined");
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   514
        } else if (expression == null) {
16256
f2d9a0c49914 8007002: Replace implicit exception throwing methods with explicit throws - simplify control flow and remove useless code
lagergren
parents: 16240
diff changeset
   515
            throw typeError(global, "cant.apply.with.to.null");
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   516
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   517
25247
c0f911459863 8046013: TypeError: Cannot apply "with" to non script object
sundar
parents: 24762
diff changeset
   518
        if (expression instanceof ScriptObjectMirror) {
c0f911459863 8046013: TypeError: Cannot apply "with" to non script object
sundar
parents: 24762
diff changeset
   519
            final Object unwrapped = ScriptObjectMirror.unwrap(expression, global);
c0f911459863 8046013: TypeError: Cannot apply "with" to non script object
sundar
parents: 24762
diff changeset
   520
            if (unwrapped instanceof ScriptObject) {
c0f911459863 8046013: TypeError: Cannot apply "with" to non script object
sundar
parents: 24762
diff changeset
   521
                return new WithObject(scope, (ScriptObject)unwrapped);
c0f911459863 8046013: TypeError: Cannot apply "with" to non script object
sundar
parents: 24762
diff changeset
   522
            }
25821
fbb51e67d2a7 8048869: Reduce compile time by about 5% by removing the Class.casts from the AST nodes
lagergren
parents: 25250
diff changeset
   523
            // foreign ScriptObjectMirror
fbb51e67d2a7 8048869: Reduce compile time by about 5% by removing the Class.casts from the AST nodes
lagergren
parents: 25250
diff changeset
   524
            final ScriptObject exprObj = global.newObject();
fbb51e67d2a7 8048869: Reduce compile time by about 5% by removing the Class.casts from the AST nodes
lagergren
parents: 25250
diff changeset
   525
            NativeObject.bindAllProperties(exprObj, (ScriptObjectMirror)expression);
fbb51e67d2a7 8048869: Reduce compile time by about 5% by removing the Class.casts from the AST nodes
lagergren
parents: 25250
diff changeset
   526
            return new WithObject(scope, exprObj);
fbb51e67d2a7 8048869: Reduce compile time by about 5% by removing the Class.casts from the AST nodes
lagergren
parents: 25250
diff changeset
   527
        }
fbb51e67d2a7 8048869: Reduce compile time by about 5% by removing the Class.casts from the AST nodes
lagergren
parents: 25250
diff changeset
   528
fbb51e67d2a7 8048869: Reduce compile time by about 5% by removing the Class.casts from the AST nodes
lagergren
parents: 25250
diff changeset
   529
        final Object wrappedExpr = JSType.toScriptObject(global, expression);
fbb51e67d2a7 8048869: Reduce compile time by about 5% by removing the Class.casts from the AST nodes
lagergren
parents: 25250
diff changeset
   530
        if (wrappedExpr instanceof ScriptObject) {
fbb51e67d2a7 8048869: Reduce compile time by about 5% by removing the Class.casts from the AST nodes
lagergren
parents: 25250
diff changeset
   531
            return new WithObject(scope, (ScriptObject)wrappedExpr);
19884
1bacbaa1bfc7 8024180: Incorrect handling of expression and parent scope in 'with' statements
sundar
parents: 19638
diff changeset
   532
        }
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   533
19884
1bacbaa1bfc7 8024180: Incorrect handling of expression and parent scope in 'with' statements
sundar
parents: 19638
diff changeset
   534
        throw typeError(global, "cant.apply.with.to.non.scriptobject");
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   535
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   536
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   537
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   538
     * ECMA 11.6.1 - The addition operator (+) - generic implementation
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   539
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   540
     * @param x  first term
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   541
     * @param y  second term
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   542
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   543
     * @return result of addition
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   544
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   545
    public static Object ADD(final Object x, final Object y) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   546
        // This prefix code to handle Number special is for optimization.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   547
        final boolean xIsNumber = x instanceof Number;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   548
        final boolean yIsNumber = y instanceof Number;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   549
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   550
        if (xIsNumber && yIsNumber) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   551
             return ((Number)x).doubleValue() + ((Number)y).doubleValue();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   552
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   553
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   554
        final boolean xIsUndefined = x == UNDEFINED;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   555
        final boolean yIsUndefined = y == UNDEFINED;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   556
24725
7bb1f687a852 8033334: Make sure that scope depth information is maintained in the RecompilableScriptFunctionDatas, to avoid unnecessary slow proto linkage when doing on demand compilation
lagergren
parents: 24719
diff changeset
   557
        if (xIsNumber && yIsUndefined || xIsUndefined && yIsNumber || xIsUndefined && yIsUndefined) {
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   558
            return Double.NaN;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   559
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   560
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   561
        // code below is as per the spec.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   562
        final Object xPrim = JSType.toPrimitive(x);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   563
        final Object yPrim = JSType.toPrimitive(y);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   564
29282
a8523237b66c 8074031: Canonicalize is-a-JS-string tests
attila
parents: 29281
diff changeset
   565
        if (isString(xPrim) || isString(yPrim)) {
25250
b5a4e0ac31d1 8047359: large string size RangeError should be thrown rather than reporting negative length
sundar
parents: 25247
diff changeset
   566
            try {
b5a4e0ac31d1 8047359: large string size RangeError should be thrown rather than reporting negative length
sundar
parents: 25247
diff changeset
   567
                return new ConsString(JSType.toCharSequence(xPrim), JSType.toCharSequence(yPrim));
b5a4e0ac31d1 8047359: large string size RangeError should be thrown rather than reporting negative length
sundar
parents: 25247
diff changeset
   568
            } catch (final IllegalArgumentException iae) {
b5a4e0ac31d1 8047359: large string size RangeError should be thrown rather than reporting negative length
sundar
parents: 25247
diff changeset
   569
                throw rangeError(iae, "concat.string.too.big");
b5a4e0ac31d1 8047359: large string size RangeError should be thrown rather than reporting negative length
sundar
parents: 25247
diff changeset
   570
            }
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   571
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   572
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   573
        return JSType.toNumber(xPrim) + JSType.toNumber(yPrim);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   574
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   575
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   576
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   577
     * Debugger hook.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   578
     * TODO: currently unimplemented
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   579
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   580
     * @return undefined
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   581
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   582
    public static Object DEBUGGER() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   583
        return UNDEFINED;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   584
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   585
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   586
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   587
     * New hook
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   588
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   589
     * @param clazz type for the clss
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   590
     * @param args  constructor arguments
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   591
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   592
     * @return undefined
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   593
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   594
    public static Object NEW(final Object clazz, final Object... args) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   595
        return UNDEFINED;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   596
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   597
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   598
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   599
     * ECMA 11.4.3 The typeof Operator - generic implementation
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   600
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   601
     * @param object   the object from which to retrieve property to type check
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   602
     * @param property property in object to check
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   603
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   604
     * @return type name
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   605
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   606
    public static Object TYPEOF(final Object object, final Object property) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   607
        Object obj = object;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   608
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   609
        if (property != null) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   610
            if (obj instanceof ScriptObject) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   611
                obj = ((ScriptObject)obj).get(property);
24719
f726e9d67629 8035820: Optimistic recompilation
attila
parents: 21446
diff changeset
   612
                if(Global.isLocationPropertyPlaceholder(obj)) {
f726e9d67629 8035820: Optimistic recompilation
attila
parents: 21446
diff changeset
   613
                    if(CompilerConstants.__LINE__.name().equals(property)) {
f726e9d67629 8035820: Optimistic recompilation
attila
parents: 21446
diff changeset
   614
                        obj = Integer.valueOf(0);
f726e9d67629 8035820: Optimistic recompilation
attila
parents: 21446
diff changeset
   615
                    } else {
f726e9d67629 8035820: Optimistic recompilation
attila
parents: 21446
diff changeset
   616
                        obj = "";
f726e9d67629 8035820: Optimistic recompilation
attila
parents: 21446
diff changeset
   617
                    }
f726e9d67629 8035820: Optimistic recompilation
attila
parents: 21446
diff changeset
   618
                }
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   619
            } else if (object instanceof Undefined) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   620
                obj = ((Undefined)obj).get(property);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   621
            } else if (object == null) {
16256
f2d9a0c49914 8007002: Replace implicit exception throwing methods with explicit throws - simplify control flow and remove useless code
lagergren
parents: 16240
diff changeset
   622
                throw typeError("cant.get.property", safeToString(property), "null");
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   623
            } else if (JSType.isPrimitive(obj)) {
16188
d6390b0ea32a 8006678: Avoid too many Context.getGlobal() calls
sundar
parents: 16151
diff changeset
   624
                obj = ((ScriptObject)JSType.toScriptObject(obj)).get(property);
19889
63af9358d0dc 8024615: Refactor ScriptObjectMirror and JSObject to support external JSObject implementations
sundar
parents: 19884
diff changeset
   625
            } else if (obj instanceof JSObject) {
63af9358d0dc 8024615: Refactor ScriptObjectMirror and JSObject to support external JSObject implementations
sundar
parents: 19884
diff changeset
   626
                obj = ((JSObject)obj).getMember(property.toString());
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   627
            } else {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   628
                obj = UNDEFINED;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   629
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   630
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   631
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   632
        return JSType.of(obj).typeName();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   633
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   634
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   635
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   636
     * Throw ReferenceError when LHS of assignment or increment/decrement
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   637
     * operator is not an assignable node (say a literal)
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   638
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   639
     * @param lhs Evaluated LHS
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   640
     * @param rhs Evaluated RHS
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   641
     * @param msg Additional LHS info for error message
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   642
     * @return undefined
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   643
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   644
    public static Object REFERENCE_ERROR(final Object lhs, final Object rhs, final Object msg) {
16256
f2d9a0c49914 8007002: Replace implicit exception throwing methods with explicit throws - simplify control flow and remove useless code
lagergren
parents: 16240
diff changeset
   645
        throw referenceError("cant.be.used.as.lhs", Objects.toString(msg));
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   646
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   647
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   648
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   649
     * ECMA 11.4.1 - delete operation, generic implementation
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   650
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   651
     * @param obj       object with property to delete
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   652
     * @param property  property to delete
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   653
     * @param strict    are we in strict mode
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   654
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   655
     * @return true if property was successfully found and deleted
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   656
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   657
    public static boolean DELETE(final Object obj, final Object property, final Object strict) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   658
        if (obj instanceof ScriptObject) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   659
            return ((ScriptObject)obj).delete(property, Boolean.TRUE.equals(strict));
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   660
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   661
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   662
        if (obj instanceof Undefined) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   663
            return ((Undefined)obj).delete(property, false);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   664
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   665
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   666
        if (obj == null) {
16256
f2d9a0c49914 8007002: Replace implicit exception throwing methods with explicit throws - simplify control flow and remove useless code
lagergren
parents: 16240
diff changeset
   667
            throw typeError("cant.delete.property", safeToString(property), "null");
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   668
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   669
17978
750d0582d8e2 8015830: Javascript mapping of ScriptEngine bindings does not expose keys
sundar
parents: 17778
diff changeset
   670
        if (obj instanceof ScriptObjectMirror) {
750d0582d8e2 8015830: Javascript mapping of ScriptEngine bindings does not expose keys
sundar
parents: 17778
diff changeset
   671
            return ((ScriptObjectMirror)obj).delete(property);
750d0582d8e2 8015830: Javascript mapping of ScriptEngine bindings does not expose keys
sundar
parents: 17778
diff changeset
   672
        }
750d0582d8e2 8015830: Javascript mapping of ScriptEngine bindings does not expose keys
sundar
parents: 17778
diff changeset
   673
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   674
        if (JSType.isPrimitive(obj)) {
16188
d6390b0ea32a 8006678: Avoid too many Context.getGlobal() calls
sundar
parents: 16151
diff changeset
   675
            return ((ScriptObject) JSType.toScriptObject(obj)).delete(property, Boolean.TRUE.equals(strict));
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   676
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   677
19889
63af9358d0dc 8024615: Refactor ScriptObjectMirror and JSObject to support external JSObject implementations
sundar
parents: 19884
diff changeset
   678
        if (obj instanceof JSObject) {
63af9358d0dc 8024615: Refactor ScriptObjectMirror and JSObject to support external JSObject implementations
sundar
parents: 19884
diff changeset
   679
            ((JSObject)obj).removeMember(Objects.toString(property));
63af9358d0dc 8024615: Refactor ScriptObjectMirror and JSObject to support external JSObject implementations
sundar
parents: 19884
diff changeset
   680
            return true;
63af9358d0dc 8024615: Refactor ScriptObjectMirror and JSObject to support external JSObject implementations
sundar
parents: 19884
diff changeset
   681
        }
63af9358d0dc 8024615: Refactor ScriptObjectMirror and JSObject to support external JSObject implementations
sundar
parents: 19884
diff changeset
   682
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   683
        // if object is not reference type, vacuously delete is successful.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   684
        return true;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   685
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   686
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   687
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   688
     * ECMA 11.4.1 - delete operator, special case
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   689
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   690
     * This is 'delete' that always fails. We have to check strict mode and throw error.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   691
     * That is why this is a runtime function. Or else we could have inlined 'false'.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   692
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   693
     * @param property  property to delete
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   694
     * @param strict    are we in strict mode
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   695
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   696
     * @return false always
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   697
     */
16206
83069fa0935b 8006529: Methods always get callee - it should be conditional
attila
parents: 16188
diff changeset
   698
    public static boolean FAIL_DELETE(final Object property, final Object strict) {
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   699
        if (Boolean.TRUE.equals(strict)) {
16256
f2d9a0c49914 8007002: Replace implicit exception throwing methods with explicit throws - simplify control flow and remove useless code
lagergren
parents: 16240
diff changeset
   700
            throw syntaxError("strict.cant.delete", safeToString(property));
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   701
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   702
        return false;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   703
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   704
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   705
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   706
     * ECMA 11.9.1 - The equals operator (==) - generic implementation
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   707
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   708
     * @param x first object to compare
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   709
     * @param y second object to compare
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   710
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   711
     * @return true if type coerced versions of objects are equal
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   712
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   713
    public static boolean EQ(final Object x, final Object y) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   714
        return equals(x, y);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   715
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   716
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   717
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   718
     * ECMA 11.9.2 - The does-not-equal operator (==) - generic implementation
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   719
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   720
     * @param x first object to compare
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   721
     * @param y second object to compare
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   722
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   723
     * @return true if type coerced versions of objects are not equal
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   724
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   725
    public static boolean NE(final Object x, final Object y) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   726
        return !EQ(x, y);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   727
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   728
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   729
    /** ECMA 11.9.3 The Abstract Equality Comparison Algorithm */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   730
    private static boolean equals(final Object x, final Object y) {
24762
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   731
        if (x == y) {
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   732
            return true;
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   733
        }
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   734
        if (x instanceof ScriptObject && y instanceof ScriptObject) {
29133
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   735
            return false; // x != y
24762
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   736
        }
26052
41d18e9e45a4 8053910: ScriptObjectMirror causing havoc with Invocation interface
sundar
parents: 25828
diff changeset
   737
        if (x instanceof ScriptObjectMirror || y instanceof ScriptObjectMirror) {
41d18e9e45a4 8053910: ScriptObjectMirror causing havoc with Invocation interface
sundar
parents: 25828
diff changeset
   738
            return ScriptObjectMirror.identical(x, y);
41d18e9e45a4 8053910: ScriptObjectMirror causing havoc with Invocation interface
sundar
parents: 25828
diff changeset
   739
        }
24762
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   740
        return equalValues(x, y);
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   741
    }
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   742
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   743
    /**
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   744
     * Extracted portion of {@code equals()} that compares objects by value (or by reference, if no known value
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   745
     * comparison applies).
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   746
     * @param x one value
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   747
     * @param y another value
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   748
     * @return true if they're equal according to 11.9.3
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   749
     */
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   750
    private static boolean equalValues(final Object x, final Object y) {
24752
c835f368e8e0 8043002: Improve performance of Nashorn equality operators
attila
parents: 24745
diff changeset
   751
        final JSType xType = JSType.ofNoFunction(x);
c835f368e8e0 8043002: Improve performance of Nashorn equality operators
attila
parents: 24745
diff changeset
   752
        final JSType yType = JSType.ofNoFunction(y);
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   753
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   754
        if (xType == yType) {
24762
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   755
            return equalSameTypeValues(x, y, xType);
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   756
        }
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   757
24762
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   758
        return equalDifferentTypeValues(x, y, xType, yType);
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   759
    }
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   760
24762
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   761
    /**
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   762
     * Extracted portion of {@link #equals(Object, Object)} and {@link #strictEquals(Object, Object)} that compares
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   763
     * values belonging to the same JSType.
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   764
     * @param x one value
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   765
     * @param y another value
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   766
     * @param type the common type for the values
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   767
     * @return true if they're equal
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   768
     */
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   769
    private static boolean equalSameTypeValues(final Object x, final Object y, final JSType type) {
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   770
        if (type == JSType.UNDEFINED || type == JSType.NULL) {
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   771
            return true;
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   772
        }
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   773
24762
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   774
        if (type == JSType.NUMBER) {
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   775
            return ((Number)x).doubleValue() == ((Number)y).doubleValue();
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   776
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   777
24762
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   778
        if (type == JSType.STRING) {
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   779
            // String may be represented by ConsString
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   780
            return x.toString().equals(y.toString());
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   781
        }
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   782
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   783
        if (type == JSType.BOOLEAN) {
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   784
            return ((Boolean)x).booleanValue() == ((Boolean)y).booleanValue();
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   785
        }
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   786
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   787
        return x == y;
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   788
    }
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   789
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   790
    /**
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   791
     * Extracted portion of {@link #equals(Object, Object)} that compares values belonging to different JSTypes.
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   792
     * @param x one value
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   793
     * @param y another value
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   794
     * @param xType the type for the value x
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   795
     * @param yType the type for the value y
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   796
     * @return true if they're equal
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   797
     */
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   798
    private static boolean equalDifferentTypeValues(final Object x, final Object y, final JSType xType, final JSType yType) {
29133
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   799
        if (isUndefinedAndNull(xType, yType) || isUndefinedAndNull(yType, xType)) {
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   800
            return true;
29133
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   801
        } else if (isNumberAndString(xType, yType)) {
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   802
            return equalNumberToString(x, y);
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   803
        } else if (isNumberAndString(yType, xType)) {
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   804
            // Can reverse order as both are primitives
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   805
            return equalNumberToString(y, x);
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   806
        } else if (xType == JSType.BOOLEAN) {
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   807
            return equalBooleanToAny(x, y);
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   808
        } else if (yType == JSType.BOOLEAN) {
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   809
            // Can reverse order as y is primitive
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   810
            return equalBooleanToAny(y, x);
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   811
        } else if (isNumberOrStringAndObject(xType, yType)) {
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   812
            return equalNumberOrStringToObject(x, y);
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   813
        } else if (isNumberOrStringAndObject(yType, xType)) {
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   814
            // Can reverse order as y is primitive
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   815
            return equalNumberOrStringToObject(y, x);
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   816
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   817
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   818
        return false;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   819
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   820
29133
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   821
    private static boolean isUndefinedAndNull(final JSType xType, final JSType yType) {
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   822
        return xType == JSType.UNDEFINED && yType == JSType.NULL;
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   823
    }
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   824
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   825
    private static boolean isNumberAndString(final JSType xType, final JSType yType) {
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   826
        return xType == JSType.NUMBER && yType == JSType.STRING;
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   827
    }
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   828
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   829
    private static boolean isNumberOrStringAndObject(final JSType xType, final JSType yType) {
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   830
        return (xType == JSType.NUMBER || xType == JSType.STRING) && yType == JSType.OBJECT;
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   831
    }
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   832
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   833
    private static boolean equalNumberToString(final Object num, final Object str) {
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   834
        // Specification says comparing a number to string should be done as "equals(num, JSType.toNumber(str))". We
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   835
        // can short circuit it to this as we know that "num" is a number, so it'll end up being a number-number
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   836
        // comparison.
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   837
        return ((Number)num).doubleValue() == JSType.toNumber(str.toString());
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   838
    }
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   839
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   840
    private static boolean equalBooleanToAny(final Object bool, final Object any) {
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   841
        return equals(JSType.toNumber((Boolean)bool), any);
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   842
    }
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   843
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   844
    private static boolean equalNumberOrStringToObject(final Object numOrStr, final Object any) {
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   845
        return equals(numOrStr, JSType.toPrimitive(any));
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   846
    }
1cd7d8af99ba 8072426: Can't compare Java objects to strings or numbers
attila
parents: 26768
diff changeset
   847
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   848
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   849
     * ECMA 11.9.4 - The strict equal operator (===) - generic implementation
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   850
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   851
     * @param x first object to compare
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   852
     * @param y second object to compare
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   853
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   854
     * @return true if objects are equal
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   855
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   856
    public static boolean EQ_STRICT(final Object x, final Object y) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   857
        return strictEquals(x, y);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   858
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   859
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   860
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   861
     * ECMA 11.9.5 - The strict non equal operator (!==) - generic implementation
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   862
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   863
     * @param x first object to compare
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   864
     * @param y second object to compare
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   865
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   866
     * @return true if objects are not equal
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   867
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   868
    public static boolean NE_STRICT(final Object x, final Object y) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   869
        return !EQ_STRICT(x, y);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   870
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   871
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   872
    /** ECMA 11.9.6 The Strict Equality Comparison Algorithm */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   873
    private static boolean strictEquals(final Object x, final Object y) {
25828
077046a5d726 8044786: Some tests fail with non-optimistic compilation
attila
parents: 25821
diff changeset
   874
        // NOTE: you might be tempted to do a quick x == y comparison. Remember, though, that any Double object having
077046a5d726 8044786: Some tests fail with non-optimistic compilation
attila
parents: 25821
diff changeset
   875
        // NaN value is not equal to itself by value even though it is referentially.
24762
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   876
24752
c835f368e8e0 8043002: Improve performance of Nashorn equality operators
attila
parents: 24745
diff changeset
   877
        final JSType xType = JSType.ofNoFunction(x);
c835f368e8e0 8043002: Improve performance of Nashorn equality operators
attila
parents: 24745
diff changeset
   878
        final JSType yType = JSType.ofNoFunction(y);
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   879
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   880
        if (xType != yType) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   881
            return false;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   882
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   883
24762
96a251e57232 8043608: Make equality tests inline better
attila
parents: 24752
diff changeset
   884
        return equalSameTypeValues(x, y, xType);
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   885
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   886
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   887
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   888
     * ECMA 11.8.6 - The in operator - generic implementation
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   889
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   890
     * @param property property to check for
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   891
     * @param obj object in which to check for property
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   892
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   893
     * @return true if objects are equal
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   894
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   895
    public static boolean IN(final Object property, final Object obj) {
24752
c835f368e8e0 8043002: Improve performance of Nashorn equality operators
attila
parents: 24745
diff changeset
   896
        final JSType rvalType = JSType.ofNoFunction(obj);
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   897
24752
c835f368e8e0 8043002: Improve performance of Nashorn equality operators
attila
parents: 24745
diff changeset
   898
        if (rvalType == JSType.OBJECT) {
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   899
            if (obj instanceof ScriptObject) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   900
                return ((ScriptObject)obj).has(property);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   901
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   902
19889
63af9358d0dc 8024615: Refactor ScriptObjectMirror and JSObject to support external JSObject implementations
sundar
parents: 19884
diff changeset
   903
            if (obj instanceof JSObject) {
63af9358d0dc 8024615: Refactor ScriptObjectMirror and JSObject to support external JSObject implementations
sundar
parents: 19884
diff changeset
   904
                return ((JSObject)obj).hasMember(Objects.toString(property));
63af9358d0dc 8024615: Refactor ScriptObjectMirror and JSObject to support external JSObject implementations
sundar
parents: 19884
diff changeset
   905
            }
63af9358d0dc 8024615: Refactor ScriptObjectMirror and JSObject to support external JSObject implementations
sundar
parents: 19884
diff changeset
   906
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   907
            return false;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   908
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   909
17778
991ccffbeb13 8015459: Octane test run fails on Turkish locale
sundar
parents: 16272
diff changeset
   910
        throw typeError("in.with.non.object", rvalType.toString().toLowerCase(Locale.ENGLISH));
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   911
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   912
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   913
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   914
     * ECMA 11.8.6 - The strict instanceof operator - generic implementation
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   915
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   916
     * @param obj first object to compare
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   917
     * @param clazz type to check against
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   918
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   919
     * @return true if {@code obj} is an instanceof {@code clazz}
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   920
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   921
    public static boolean INSTANCEOF(final Object obj, final Object clazz) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   922
        if (clazz instanceof ScriptFunction) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   923
            if (obj instanceof ScriptObject) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   924
                return ((ScriptObject)clazz).isInstance((ScriptObject)obj);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   925
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   926
            return false;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   927
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   928
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   929
        if (clazz instanceof StaticClass) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   930
            return ((StaticClass)clazz).getRepresentedClass().isInstance(obj);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   931
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   932
19889
63af9358d0dc 8024615: Refactor ScriptObjectMirror and JSObject to support external JSObject implementations
sundar
parents: 19884
diff changeset
   933
        if (clazz instanceof JSObject) {
63af9358d0dc 8024615: Refactor ScriptObjectMirror and JSObject to support external JSObject implementations
sundar
parents: 19884
diff changeset
   934
            return ((JSObject)clazz).isInstance(obj);
63af9358d0dc 8024615: Refactor ScriptObjectMirror and JSObject to support external JSObject implementations
sundar
parents: 19884
diff changeset
   935
        }
63af9358d0dc 8024615: Refactor ScriptObjectMirror and JSObject to support external JSObject implementations
sundar
parents: 19884
diff changeset
   936
63af9358d0dc 8024615: Refactor ScriptObjectMirror and JSObject to support external JSObject implementations
sundar
parents: 19884
diff changeset
   937
        // provide for reverse hook
63af9358d0dc 8024615: Refactor ScriptObjectMirror and JSObject to support external JSObject implementations
sundar
parents: 19884
diff changeset
   938
        if (obj instanceof JSObject) {
63af9358d0dc 8024615: Refactor ScriptObjectMirror and JSObject to support external JSObject implementations
sundar
parents: 19884
diff changeset
   939
            return ((JSObject)obj).isInstanceOf(clazz);
18334
47413e8d71b5 8016618: script mirror object access should be improved
sundar
parents: 17978
diff changeset
   940
        }
47413e8d71b5 8016618: script mirror object access should be improved
sundar
parents: 17978
diff changeset
   941
16256
f2d9a0c49914 8007002: Replace implicit exception throwing methods with explicit throws - simplify control flow and remove useless code
lagergren
parents: 16240
diff changeset
   942
        throw typeError("instanceof.on.non.object");
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   943
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   944
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   945
    /**
16272
675a0caf75bc 8009263: Fix all javadoc errors in nashorn code
sundar
parents: 16262
diff changeset
   946
     * ECMA 11.8.1 - The less than operator ({@literal <}) - generic implementation
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   947
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   948
     * @param x first object to compare
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   949
     * @param y second object to compare
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   950
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   951
     * @return true if x is less than y
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   952
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   953
    public static boolean LT(final Object x, final Object y) {
29283
fb47e4d25a9f 8035712: Restore some of the RuntimeCallSite specializations
attila
parents: 29282
diff changeset
   954
        final Object px = JSType.toPrimitive(x, Number.class);
fb47e4d25a9f 8035712: Restore some of the RuntimeCallSite specializations
attila
parents: 29282
diff changeset
   955
        final Object py = JSType.toPrimitive(y, Number.class);
fb47e4d25a9f 8035712: Restore some of the RuntimeCallSite specializations
attila
parents: 29282
diff changeset
   956
fb47e4d25a9f 8035712: Restore some of the RuntimeCallSite specializations
attila
parents: 29282
diff changeset
   957
        return areBothString(px, py) ? px.toString().compareTo(py.toString()) < 0 :
fb47e4d25a9f 8035712: Restore some of the RuntimeCallSite specializations
attila
parents: 29282
diff changeset
   958
            JSType.toNumber(px) < JSType.toNumber(py);
fb47e4d25a9f 8035712: Restore some of the RuntimeCallSite specializations
attila
parents: 29282
diff changeset
   959
    }
fb47e4d25a9f 8035712: Restore some of the RuntimeCallSite specializations
attila
parents: 29282
diff changeset
   960
fb47e4d25a9f 8035712: Restore some of the RuntimeCallSite specializations
attila
parents: 29282
diff changeset
   961
    private static boolean areBothString(final Object x, final Object y) {
fb47e4d25a9f 8035712: Restore some of the RuntimeCallSite specializations
attila
parents: 29282
diff changeset
   962
        return isString(x) && isString(y);
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   963
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   964
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   965
    /**
16272
675a0caf75bc 8009263: Fix all javadoc errors in nashorn code
sundar
parents: 16262
diff changeset
   966
     * ECMA 11.8.2 - The greater than operator ({@literal >}) - generic implementation
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   967
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   968
     * @param x first object to compare
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   969
     * @param y second object to compare
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   970
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   971
     * @return true if x is greater than y
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   972
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   973
    public static boolean GT(final Object x, final Object y) {
29283
fb47e4d25a9f 8035712: Restore some of the RuntimeCallSite specializations
attila
parents: 29282
diff changeset
   974
        final Object px = JSType.toPrimitive(x, Number.class);
fb47e4d25a9f 8035712: Restore some of the RuntimeCallSite specializations
attila
parents: 29282
diff changeset
   975
        final Object py = JSType.toPrimitive(y, Number.class);
fb47e4d25a9f 8035712: Restore some of the RuntimeCallSite specializations
attila
parents: 29282
diff changeset
   976
fb47e4d25a9f 8035712: Restore some of the RuntimeCallSite specializations
attila
parents: 29282
diff changeset
   977
        return areBothString(px, py) ? px.toString().compareTo(py.toString()) > 0 :
fb47e4d25a9f 8035712: Restore some of the RuntimeCallSite specializations
attila
parents: 29282
diff changeset
   978
            JSType.toNumber(px) > JSType.toNumber(py);
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   979
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   980
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   981
    /**
16272
675a0caf75bc 8009263: Fix all javadoc errors in nashorn code
sundar
parents: 16262
diff changeset
   982
     * ECMA 11.8.3 - The less than or equal operator ({@literal <=}) - generic implementation
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   983
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   984
     * @param x first object to compare
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   985
     * @param y second object to compare
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   986
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   987
     * @return true if x is less than or equal to y
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   988
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   989
    public static boolean LE(final Object x, final Object y) {
29283
fb47e4d25a9f 8035712: Restore some of the RuntimeCallSite specializations
attila
parents: 29282
diff changeset
   990
        final Object px = JSType.toPrimitive(x, Number.class);
fb47e4d25a9f 8035712: Restore some of the RuntimeCallSite specializations
attila
parents: 29282
diff changeset
   991
        final Object py = JSType.toPrimitive(y, Number.class);
fb47e4d25a9f 8035712: Restore some of the RuntimeCallSite specializations
attila
parents: 29282
diff changeset
   992
fb47e4d25a9f 8035712: Restore some of the RuntimeCallSite specializations
attila
parents: 29282
diff changeset
   993
        return areBothString(px, py) ? px.toString().compareTo(py.toString()) <= 0 :
fb47e4d25a9f 8035712: Restore some of the RuntimeCallSite specializations
attila
parents: 29282
diff changeset
   994
            JSType.toNumber(px) <= JSType.toNumber(py);
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   995
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   996
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   997
    /**
16272
675a0caf75bc 8009263: Fix all javadoc errors in nashorn code
sundar
parents: 16262
diff changeset
   998
     * ECMA 11.8.4 - The greater than or equal operator ({@literal >=}) - generic implementation
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   999
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
  1000
     * @param x first object to compare
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
  1001
     * @param y second object to compare
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
  1002
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
  1003
     * @return true if x is greater than or equal to y
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
  1004
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
  1005
    public static boolean GE(final Object x, final Object y) {
29283
fb47e4d25a9f 8035712: Restore some of the RuntimeCallSite specializations
attila
parents: 29282
diff changeset
  1006
        final Object px = JSType.toPrimitive(x, Number.class);
fb47e4d25a9f 8035712: Restore some of the RuntimeCallSite specializations
attila
parents: 29282
diff changeset
  1007
        final Object py = JSType.toPrimitive(y, Number.class);
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
  1008
29283
fb47e4d25a9f 8035712: Restore some of the RuntimeCallSite specializations
attila
parents: 29282
diff changeset
  1009
        return areBothString(px, py) ? px.toString().compareTo(py.toString()) >= 0 :
fb47e4d25a9f 8035712: Restore some of the RuntimeCallSite specializations
attila
parents: 29282
diff changeset
  1010
            JSType.toNumber(px) >= JSType.toNumber(py);
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
  1011
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
  1012
26768
751b0f427090 8025435: Optimistic builtins support, implemented initial optimistic versions of push, pop, and charCodeAt
lagergren
parents: 26377
diff changeset
  1013
    /**
751b0f427090 8025435: Optimistic builtins support, implemented initial optimistic versions of push, pop, and charCodeAt
lagergren
parents: 26377
diff changeset
  1014
     * Tag a reserved name as invalidated - used when someone writes
751b0f427090 8025435: Optimistic builtins support, implemented initial optimistic versions of push, pop, and charCodeAt
lagergren
parents: 26377
diff changeset
  1015
     * to a property with this name - overly conservative, but link time
751b0f427090 8025435: Optimistic builtins support, implemented initial optimistic versions of push, pop, and charCodeAt
lagergren
parents: 26377
diff changeset
  1016
     * is too late to apply e.g. apply-&gt;call specialization
751b0f427090 8025435: Optimistic builtins support, implemented initial optimistic versions of push, pop, and charCodeAt
lagergren
parents: 26377
diff changeset
  1017
     * @param name property name
751b0f427090 8025435: Optimistic builtins support, implemented initial optimistic versions of push, pop, and charCodeAt
lagergren
parents: 26377
diff changeset
  1018
     */
751b0f427090 8025435: Optimistic builtins support, implemented initial optimistic versions of push, pop, and charCodeAt
lagergren
parents: 26377
diff changeset
  1019
    public static void invalidateReservedBuiltinName(final String name) {
751b0f427090 8025435: Optimistic builtins support, implemented initial optimistic versions of push, pop, and charCodeAt
lagergren
parents: 26377
diff changeset
  1020
        final Context context = Context.getContextTrusted();
751b0f427090 8025435: Optimistic builtins support, implemented initial optimistic versions of push, pop, and charCodeAt
lagergren
parents: 26377
diff changeset
  1021
        final SwitchPoint sp = context.getBuiltinSwitchPoint(name);
751b0f427090 8025435: Optimistic builtins support, implemented initial optimistic versions of push, pop, and charCodeAt
lagergren
parents: 26377
diff changeset
  1022
        assert sp != null;
29283
fb47e4d25a9f 8035712: Restore some of the RuntimeCallSite specializations
attila
parents: 29282
diff changeset
  1023
        context.getLogger(ApplySpecialization.class).info("Overwrote special name '" + name +"' - invalidating switchpoint");
fb47e4d25a9f 8035712: Restore some of the RuntimeCallSite specializations
attila
parents: 29282
diff changeset
  1024
        SwitchPoint.invalidateAll(new SwitchPoint[] { sp });
26768
751b0f427090 8025435: Optimistic builtins support, implemented initial optimistic versions of push, pop, and charCodeAt
lagergren
parents: 26377
diff changeset
  1025
    }
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
  1026
}