nashorn/src/jdk/nashorn/internal/runtime/PropertyMap.java
author sundar
Tue, 08 Jan 2013 08:51:00 +0530
changeset 16154 de44634fa4ec
parent 16152 ea430b83d74d
child 16201 889ddb179cdf
permissions -rw-r--r--
8005782: get rid of javadoc errors, warnings in nashorn build Reviewed-by: 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.runtime.PropertyHashMap.EMPTY_MAP;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    29
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    30
import java.lang.invoke.MethodHandle;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    31
import java.lang.invoke.SwitchPoint;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    32
import java.lang.ref.WeakReference;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    33
import java.util.Arrays;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    34
import java.util.Collection;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    35
import java.util.HashMap;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    36
import java.util.Iterator;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    37
import java.util.LinkedHashMap;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    38
import java.util.Map;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    39
import java.util.NoSuchElementException;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    40
import java.util.WeakHashMap;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    41
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    42
/**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    43
 * Map of object properties. The PropertyMap is the "template" for JavaScript object
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    44
 * layouts. It contains a map with prototype names as keys and {@link Property} instances
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    45
 * as values. A PropertyMap is typically passed to the {@link ScriptObject} constructor
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    46
 * to form the seed map for the ScriptObject.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    47
 * <p>
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    48
 * All property maps are immutable. If a property is added, modified or removed, the mutator
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    49
 * will return a new map.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    50
 */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    51
public final class PropertyMap implements Iterable<Object>, PropertyListener {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    52
    /** Is this a prototype PropertyMap? */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    53
    public static final int IS_PROTOTYPE          = 0b0000_0001;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    54
    /** Used for non extensible PropertyMaps, negative logic as the normal case is extensible. See {@link ScriptObject#preventExtensions()} */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    55
    public static final int NOT_EXTENSIBLE        = 0b0000_0010;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    56
    /** This mask is used to preserve certain flags when cloning the PropertyMap. Others should not be copied */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    57
    private static final int CLONEABLE_FLAGS_MASK = 0b0000_1111;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    58
    /** Has a listener been added to this property map. This flag is not copied when cloning a map. See {@link PropertyListener} */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    59
    public static final int IS_LISTENER_ADDED     = 0b0001_0000;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    60
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    61
    /** Map status flags. */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    62
    private int flags;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    63
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    64
    /** Class of object referenced.*/
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    65
    private final Class<?> structure;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    66
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    67
    /** Context associated with this {@link PropertyMap}. */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    68
    private final Context context;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    69
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    70
    /** Map of properties. */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    71
    private final PropertyHashMap properties;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    72
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    73
    /** objects proto. */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    74
    private ScriptObject proto;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    75
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    76
    /** Length of spill in use. */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    77
    private int spillLength;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    78
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    79
    /** {@link SwitchPoint}s for gets on inherited properties. */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    80
    private Map<String, SwitchPoint> protoGetSwitches;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    81
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    82
    /** History of maps, used to limit map duplication. */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    83
    private HashMap<Property, PropertyMap> history;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    84
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    85
    /** History of prototypes, used to limit map duplication. */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    86
    private WeakHashMap<ScriptObject, WeakReference<PropertyMap>> protoHistory;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    87
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    88
    /** Cache for hashCode */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    89
    private int hashCode;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    90
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    91
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    92
     * Constructor.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    93
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    94
     * @param structure  Class the map's {@link AccessorProperty}s apply to.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    95
     * @param context    Context associated with this {@link PropertyMap}.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    96
     * @param properties A {@link PropertyHashMap} with initial contents.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    97
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    98
    PropertyMap(final Class<?> structure, final Context context, final PropertyHashMap properties) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    99
        this.structure  = structure;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   100
        this.context    = context;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   101
        this.properties = properties;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   102
        this.hashCode   = computeHashCode();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   103
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   104
        if (Context.DEBUG) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   105
            count++;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   106
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   107
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   108
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   109
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   110
     * Cloning constructor.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   111
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   112
     * @param propertyMap Existing property map.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   113
     * @param properties  A {@link PropertyHashMap} with a new set of properties.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   114
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   115
    private PropertyMap(final PropertyMap propertyMap, final PropertyHashMap properties) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   116
        this.structure   = propertyMap.structure;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   117
        this.context     = propertyMap.context;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   118
        this.properties  = properties;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   119
        this.flags       = propertyMap.getClonedFlags();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   120
        this.proto       = propertyMap.proto;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   121
        this.spillLength = propertyMap.spillLength;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   122
        this.hashCode    = computeHashCode();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   123
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   124
        if (Context.DEBUG) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   125
            count++;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   126
            clonedCount++;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   127
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   128
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   129
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   130
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   131
     * Duplicates this PropertyMap instance. This is used by nasgen generated
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   132
     * prototype and constructor classes. {@link PropertyMap} used for singletons
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   133
     * like these (and global instance) are duplicated using this method and used.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   134
     * The original filled map referenced by static fields of prototype and
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   135
     * constructor classes are not touched. This allows multiple independent global
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   136
     * instances to be used within a single context instance.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   137
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   138
     * @return Duplicated {@link PropertyMap}.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   139
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   140
    public PropertyMap duplicate() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   141
        return new PropertyMap(this.structure, this.context, this.properties);
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
     * Public property map allocator.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   146
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   147
     * @param structure  Class the map's {@link AccessorProperty}s apply to.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   148
     * @param properties Collection of initial properties.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   149
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   150
     * @return New {@link PropertyMap}.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   151
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   152
    public static PropertyMap newMap(final Class<?> structure, final Collection<Property> properties) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   153
        final Context context = Context.fromClass(structure);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   154
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   155
        // Reduce the number of empty maps in the context.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   156
        if (structure == jdk.nashorn.internal.scripts.JO$.class) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   157
            return context.emptyMap;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   158
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   159
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   160
        PropertyHashMap newProperties = EMPTY_MAP.immutableAdd(properties);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   161
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   162
        return new PropertyMap(structure, context, newProperties);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   163
    }
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
     * Public property map factory allocator
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   167
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   168
     * @param structure  Class the map's {@link AccessorProperty}s apply to.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   169
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   170
     * @return New {@link PropertyMap}.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   171
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   172
    public static PropertyMap newMap(final Class<?> structure) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   173
        return newMap(structure, null);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   174
    }
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
     * Return a sharable empty map.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   178
     *
16152
ea430b83d74d 8005789: Forgot to document -Dnashorn.unstable.relink.threshold
lagergren
parents: 16151
diff changeset
   179
     * @param  context the context
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   180
     * @return New empty {@link PropertyMap}.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   181
     */
16152
ea430b83d74d 8005789: Forgot to document -Dnashorn.unstable.relink.threshold
lagergren
parents: 16151
diff changeset
   182
    public static PropertyMap newEmptyMap(final Context context) {
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   183
        return new PropertyMap(jdk.nashorn.internal.scripts.JO$.class, context, EMPTY_MAP);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   184
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   185
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   186
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   187
     * Return number of properties in the map.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   188
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   189
     * @return Number of properties.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   190
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   191
    public int size() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   192
        return properties.size();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   193
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   194
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   195
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   196
     * Return a SwitchPoint used to track changes of a property in a prototype.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   197
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   198
     * @param key {@link Property} key.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   199
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   200
     * @return A shared {@link SwitchPoint} for the property.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   201
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   202
    public SwitchPoint getProtoGetSwitchPoint(final String key) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   203
        if (proto == null) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   204
            return null;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   205
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   206
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   207
        if (protoGetSwitches == null) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   208
            protoGetSwitches = new HashMap<>();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   209
            if (! isListenerAdded()) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   210
                proto.addPropertyListener(this);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   211
                setIsListenerAdded();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   212
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   213
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   214
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   215
        if (protoGetSwitches.containsKey(key)) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   216
            return protoGetSwitches.get(key);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   217
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   218
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   219
        final SwitchPoint switchPoint = new SwitchPoint();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   220
        protoGetSwitches.put(key, switchPoint);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   221
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   222
        return switchPoint;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   223
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   224
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   225
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   226
     * Indicate that a prototype property hash changed.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   227
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   228
     * @param property {@link Property} to invalidate.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   229
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   230
    private void invalidateProtoGetSwitchPoint(final Property property) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   231
        if (protoGetSwitches != null) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   232
            final String key = property.getKey();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   233
            final SwitchPoint sp = protoGetSwitches.get(key);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   234
            if (sp != null) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   235
                protoGetSwitches.put(key, new SwitchPoint());
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   236
                if (Context.DEBUG) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   237
                    protoInvalidations++;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   238
                }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   239
                SwitchPoint.invalidateAll(new SwitchPoint[] { sp });
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   240
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   241
        }
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
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   245
     * Add a property to the map.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   246
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   247
     * @param property {@link Property} being added.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   248
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   249
     * @return New {@link PropertyMap} with {@link Property} added.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   250
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   251
    public PropertyMap newProperty(final Property property) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   252
        return addProperty(property);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   253
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   254
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   255
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   256
     * Add a property to the map, re-binding its getters and setters,
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   257
     * if available, to a given receiver. This is typically the global scope. See
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   258
     * {@link ScriptObject#addBoundProperties(ScriptObject)}
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   259
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   260
     * @param property {@link Property} being added.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   261
     * @param bindTo   Object to bind to.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   262
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   263
     * @return New {@link PropertyMap} with {@link Property} added.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   264
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   265
    PropertyMap newPropertyBind(final AccessorProperty property, final ScriptObject bindTo) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   266
        return newProperty(new AccessorProperty(property, bindTo));
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   267
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   268
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   269
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   270
     * Add a new accessor property to the map.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   271
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   272
     * @param key           {@link Property} key.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   273
     * @param propertyFlags {@link Property} flags.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   274
     * @param getter        {@link Property} get accessor method.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   275
     * @param setter        {@link Property} set accessor method.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   276
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   277
     * @return  New {@link PropertyMap} with {@link AccessorProperty} added.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   278
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   279
    public PropertyMap newProperty(final String key, final int propertyFlags, final MethodHandle getter, final MethodHandle setter) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   280
        return newProperty(new AccessorProperty(key, propertyFlags, getter, setter));
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   281
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   282
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   283
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   284
     * Add a property to the map.  Cloning or using an existing map if available.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   285
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   286
     * @param property {@link Property} being added.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   287
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   288
     * @return New {@link PropertyMap} with {@link Property} added.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   289
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   290
    PropertyMap addProperty(final Property property) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   291
        PropertyMap newMap = checkHistory(property);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   292
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   293
        if (newMap == null) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   294
            final PropertyHashMap newProperties = properties.immutableAdd(property);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   295
            newMap = new PropertyMap(this, newProperties);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   296
            addToHistory(property, newMap);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   297
            newMap.spillLength += property.getSpillCount();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   298
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   299
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   300
        return newMap;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   301
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   302
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   303
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   304
     * Remove a property from a map. Cloning or using an existing map if available.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   305
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   306
     * @param property {@link Property} being removed.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   307
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   308
     * @return New {@link PropertyMap} with {@link Property} removed or {@code null} if not found.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   309
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   310
    PropertyMap deleteProperty(final Property property) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   311
        PropertyMap newMap = checkHistory(property);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   312
        final String key = property.getKey();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   313
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   314
        if (newMap == null && properties.containsKey(key)) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   315
            final PropertyHashMap newProperties = properties.immutableRemove(key);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   316
            newMap = new PropertyMap(this, newProperties);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   317
            addToHistory(property, newMap);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   318
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   319
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   320
        return newMap;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   321
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   322
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   323
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   324
     * Replace an existing property with a new one.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   325
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   326
     * @param oldProperty Property to replace.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   327
     * @param newProperty New {@link Property}.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   328
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   329
     * @return New {@link PropertyMap} with {@link Property} replaced.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   330
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   331
    PropertyMap replaceProperty(final Property oldProperty, final Property newProperty) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   332
        // Add replaces existing property.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   333
        final PropertyHashMap newProperties = properties.immutableAdd(newProperty);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   334
        final PropertyMap newMap = new PropertyMap(this, newProperties);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   335
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   336
        /*
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   337
         * See ScriptObject.modifyProperty and ScriptObject.setUserAccessors methods.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   338
         *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   339
         * This replaceProperty method is called only for the following three cases:
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   340
         *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   341
         *   1. To change flags OR TYPE of an old (cloned) property. We use the same spill slots.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   342
         *   2. To change one UserAccessor property with another - user getter or setter changed via
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   343
         *      Object.defineProperty function. Again, same spill slots are re-used.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   344
         *   3. Via ScriptObject.setUserAccessors method to set user getter and setter functions
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   345
         *      replacing the dummy AccessorProperty with null method handles (added during map init).
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   346
         *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   347
         * In case (1) and case(2), the property type of old and new property is same. For case (3),
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   348
         * the old property is an AccessorProperty and the new one is a UserAccessorProperty property.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   349
         */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   350
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   351
        final boolean sameType = (oldProperty.getClass() == newProperty.getClass());
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   352
        assert sameType ||
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   353
                (oldProperty instanceof AccessorProperty &&
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   354
                newProperty instanceof UserAccessorProperty) : "arbitrary replaceProperty attempted";
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   355
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   356
        newMap.flags = getClonedFlags();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   357
        newMap.proto = proto;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   358
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   359
        /*
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   360
         * spillLength remains same in case (1) and (2) because of slot reuse. Only for case (3), we need
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   361
         * to add spill count of the newly added UserAccessorProperty property.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   362
         */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   363
        newMap.spillLength = spillLength + (sameType? 0 : newProperty.getSpillCount());
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   364
        return newMap;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   365
    }
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
     * Find a property in the map.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   369
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   370
     * @param key Key to search for.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   371
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   372
     * @return {@link Property} matching key.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   373
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   374
    public Property findProperty(final String key) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   375
        return properties.find(key);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   376
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   377
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   378
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   379
     * Adds all map properties from another map.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   380
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   381
     * @param other The source of properties.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   382
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   383
     * @return New {@link PropertyMap} with added properties.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   384
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   385
    public PropertyMap addAll(final PropertyMap other) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   386
        final Property[] otherProperties = other.properties.getProperties();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   387
        final PropertyHashMap newProperties = properties.immutableAdd(otherProperties);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   388
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   389
        final PropertyMap newMap = new PropertyMap(this, newProperties);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   390
        for (final Property property : otherProperties) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   391
            newMap.spillLength += property.getSpillCount();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   392
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   393
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   394
        return newMap;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   395
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   396
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   397
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   398
     * Return an array of all properties.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   399
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   400
     * @return Properties as an array.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   401
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   402
    public Property[] getProperties() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   403
        return properties.getProperties();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   404
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   405
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   406
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   407
     * Prevents the map from having additional properties.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   408
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   409
     * @return New map with {@link #NOT_EXTENSIBLE} flag set.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   410
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   411
    PropertyMap preventExtensions() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   412
        final PropertyMap newMap = new PropertyMap(this, this.properties);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   413
        newMap.flags |= NOT_EXTENSIBLE;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   414
        return newMap;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   415
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   416
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   417
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   418
     * Prevents properties in map from being modified.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   419
     *
16154
de44634fa4ec 8005782: get rid of javadoc errors, warnings in nashorn build
sundar
parents: 16152
diff changeset
   420
     * @return New map with {@link #NOT_EXTENSIBLE} flag set and properties with
de44634fa4ec 8005782: get rid of javadoc errors, warnings in nashorn build
sundar
parents: 16152
diff changeset
   421
     * {@link Property#NOT_CONFIGURABLE} set.
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   422
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   423
    PropertyMap seal() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   424
        PropertyHashMap newProperties = EMPTY_MAP;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   425
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   426
        for (final Property oldProperty :  properties.getProperties()) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   427
            newProperties = newProperties.immutableAdd(oldProperty.addFlags(Property.NOT_CONFIGURABLE));
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   428
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   429
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   430
        final PropertyMap newMap = new PropertyMap(this, newProperties);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   431
        newMap.flags |= NOT_EXTENSIBLE;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   432
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   433
        return newMap;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   434
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   435
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   436
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   437
     * Prevents properties in map from being modified or written to.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   438
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   439
     * @return New map with {@link #NOT_EXTENSIBLE} flag set and properties with
16154
de44634fa4ec 8005782: get rid of javadoc errors, warnings in nashorn build
sundar
parents: 16152
diff changeset
   440
     * {@link Property#NOT_CONFIGURABLE} and {@link Property#NOT_WRITABLE} set.
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   441
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   442
    PropertyMap freeze() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   443
        PropertyHashMap newProperties = EMPTY_MAP;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   444
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   445
        for (Property oldProperty : properties.getProperties()) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   446
            int propertyFlags = Property.NOT_CONFIGURABLE;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   447
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   448
            if (!(oldProperty instanceof UserAccessorProperty)) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   449
                propertyFlags |= Property.NOT_WRITABLE;
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
            newProperties = newProperties.immutableAdd(oldProperty.addFlags(propertyFlags));
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   453
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   454
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   455
        final PropertyMap newMap = new PropertyMap(this, newProperties);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   456
        newMap.flags |= NOT_EXTENSIBLE;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   457
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   458
        return newMap;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   459
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   460
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   461
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   462
     * Check for any configurable properties.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   463
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   464
     * @return {@code true} if any configurable.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   465
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   466
    private boolean anyConfigurable() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   467
        for (final Property property : properties.getProperties()) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   468
            if (property.isConfigurable()) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   469
               return true;
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
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   473
        return false;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   474
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   475
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   476
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   477
     * Check if all properties are frozen.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   478
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   479
     * @return {@code true} if all are frozen.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   480
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   481
    private boolean allFrozen() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   482
        for (final Property property : properties.getProperties()) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   483
            // check if it is a data descriptor
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   484
            if (!(property instanceof UserAccessorProperty)) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   485
                if (property.isWritable()) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   486
                    return false;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   487
                }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   488
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   489
            if (property.isConfigurable()) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   490
               return false;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   491
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   492
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   493
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   494
        return true;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   495
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   496
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   497
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   498
     * Check prototype history for an existing property map with specified prototype.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   499
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   500
     * @param newProto New prototype object.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   501
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   502
     * @return Existing {@link PropertyMap} or {@code null} if not found.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   503
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   504
    private PropertyMap checkProtoHistory(final ScriptObject newProto) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   505
        final PropertyMap cachedMap;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   506
        if (protoHistory != null) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   507
            final WeakReference<PropertyMap> weakMap = protoHistory.get(newProto);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   508
            cachedMap = (weakMap != null ? weakMap.get() : null);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   509
        } else {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   510
            cachedMap = null;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   511
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   512
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   513
        if (Context.DEBUG && cachedMap != null) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   514
            protoHistoryHit++;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   515
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   516
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   517
        return cachedMap;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   518
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   519
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   520
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   521
     * Add a map to the prototype history.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   522
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   523
     * @param newProto Prototype to add (key.)
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   524
     * @param newMap   {@link PropertyMap} associated with prototype.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   525
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   526
    private void addToProtoHistory(final ScriptObject newProto, final PropertyMap newMap) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   527
        if (protoHistory == null) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   528
            protoHistory = new WeakHashMap<>();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   529
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   530
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   531
        protoHistory.put(newProto, new WeakReference<>(newMap));
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   532
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   533
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   534
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   535
     * Track the modification of the map.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   536
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   537
     * @param property Mapping property.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   538
     * @param newMap   Modified {@link PropertyMap}.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   539
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   540
    private void addToHistory(final Property property, final PropertyMap newMap) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   541
        if (history == null) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   542
            history = new LinkedHashMap<>();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   543
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   544
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   545
        history.put(property, newMap);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   546
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   547
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   548
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   549
     * Check the history for a map that already has the given property added.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   550
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   551
     * @param property {@link Property} to add.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   552
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   553
     * @return Existing map or {@code null} if not found.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   554
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   555
    private PropertyMap checkHistory(final Property property) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   556
        if (history != null) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   557
            PropertyMap historicMap = history.get(property);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   558
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   559
            if (historicMap != null) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   560
                if (Context.DEBUG) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   561
                    historyHit++;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   562
                }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   563
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   564
                return historicMap;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   565
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   566
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   567
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   568
        return null;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   569
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   570
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   571
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   572
     * Calculate the hash code for the map.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   573
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   574
     * @return Computed hash code.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   575
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   576
    private int computeHashCode() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   577
        int hash = structure.hashCode();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   578
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   579
        if (proto != null) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   580
            hash ^= proto.hashCode();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   581
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   582
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   583
        for (final Property property : getProperties()) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   584
            hash = hash << 7 ^ hash >> 7;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   585
            hash ^= property.hashCode();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   586
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   587
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   588
        return hash;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   589
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   590
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   591
    @Override
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   592
    public int hashCode() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   593
        return hashCode;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   594
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   595
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   596
    @Override
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   597
    public boolean equals(final Object other) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   598
        if (!(other instanceof PropertyMap)) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   599
            return false;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   600
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   601
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   602
        final PropertyMap otherMap = (PropertyMap)other;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   603
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   604
        if (structure != otherMap.structure ||
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   605
            proto != otherMap.proto ||
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   606
            properties.size() != otherMap.properties.size()) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   607
            return false;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   608
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   609
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   610
        final Iterator<Property> iter      = properties.values().iterator();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   611
        final Iterator<Property> otherIter = otherMap.properties.values().iterator();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   612
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   613
        while (iter.hasNext() && otherIter.hasNext()) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   614
            if (!iter.next().equals(otherIter.next())) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   615
                return false;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   616
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   617
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   618
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   619
        return true;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   620
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   621
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   622
    @Override
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   623
    public String toString() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   624
        final StringBuilder sb = new StringBuilder();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   625
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   626
        sb.append(" [");
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   627
        boolean isFirst = true;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   628
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   629
        for (final Property property : properties.values()) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   630
            if (!isFirst) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   631
                sb.append(", ");
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   632
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   633
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   634
            isFirst = false;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   635
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   636
            sb.append(ScriptRuntime.safeToString(property.getKey()));
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   637
            final Class<?> ctype = property.getCurrentType();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   638
            sb.append(" <" + property.getClass().getSimpleName() + ":" + (ctype == null ? "undefined" : ctype.getSimpleName()) + ">");
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   639
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   640
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   641
        sb.append(']');
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   642
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   643
        return sb.toString();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   644
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   645
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   646
    @Override
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   647
    public Iterator<Object> iterator() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   648
        return new PropertyMapIterator(this);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   649
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   650
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   651
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   652
     * Return map's {@link Context}.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   653
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   654
     * @return The {@link Context} where the map originated.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   655
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   656
    Context getContext() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   657
        return context;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   658
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   659
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   660
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   661
     * Check if this map is a prototype
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   662
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   663
     * @return {@code true} if is prototype
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   664
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   665
    public boolean isPrototype() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   666
        return (flags & IS_PROTOTYPE) != 0;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   667
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   668
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   669
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   670
     * Flag this map as having a prototype.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   671
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   672
    private void setIsPrototype() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   673
        flags |= IS_PROTOTYPE;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   674
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   675
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   676
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   677
     * Check whether a {@link PropertyListener} has been added to this map.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   678
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   679
     * @return {@code true} if {@link PropertyListener} exists
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   680
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   681
    public boolean isListenerAdded() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   682
        return (flags & IS_LISTENER_ADDED) != 0;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   683
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   684
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   685
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   686
     * Test to see if {@link PropertyMap} is extensible.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   687
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   688
     * @return {@code true} if {@link PropertyMap} can be added to.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   689
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   690
    boolean isExtensible() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   691
        return (flags & NOT_EXTENSIBLE) == 0;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   692
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   693
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   694
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   695
     * Test to see if {@link PropertyMap} is not extensible or any properties
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   696
     * can not be modified.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   697
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   698
     * @return {@code true} if {@link PropertyMap} is sealed.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   699
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   700
    boolean isSealed() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   701
        return !isExtensible() && !anyConfigurable();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   702
    }
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
     * Test to see if {@link PropertyMap} is not extensible or all properties
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   706
     * can not be modified.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   707
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   708
     * @return {@code true} if {@link PropertyMap} is frozen.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   709
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   710
    boolean isFrozen() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   711
        return !isExtensible() && allFrozen();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   712
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   713
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   714
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   715
     * Get length of spill area associated with this {@link PropertyMap}.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   716
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   717
     * @return Length of spill area.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   718
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   719
    int getSpillLength() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   720
        return spillLength;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   721
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   722
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   723
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   724
     * Return the prototype of objects associated with this {@link PropertyMap}.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   725
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   726
     * @return Prototype object.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   727
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   728
    ScriptObject getProto() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   729
        return proto;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   730
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   731
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   732
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   733
     * Set the prototype of objects associated with this {@link PropertyMap}.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   734
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   735
     * @param newProto Prototype object to use.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   736
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   737
     * @return New {@link PropertyMap} with prototype changed.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   738
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   739
    PropertyMap setProto(final ScriptObject newProto) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   740
        final ScriptObject oldProto = this.proto;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   741
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   742
        if (oldProto == newProto) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   743
            return this;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   744
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   745
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   746
        final PropertyMap nextMap = checkProtoHistory(newProto);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   747
        if (nextMap != null) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   748
            return nextMap;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   749
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   750
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   751
        if (Context.DEBUG) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   752
            incrementSetProtoNewMapCount();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   753
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   754
        final PropertyMap newMap = new PropertyMap(this, this.properties);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   755
        addToProtoHistory(newProto, newMap);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   756
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   757
        newMap.proto = newProto;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   758
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   759
        if (oldProto != null && newMap.isListenerAdded()) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   760
            oldProto.removePropertyListener(newMap);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   761
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   762
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   763
        if (newProto != null) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   764
            newProto.getMap().setIsPrototype();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   765
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   766
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   767
        return newMap;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   768
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   769
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   770
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   771
     * Indicate that the map has listeners.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   772
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   773
    private void setIsListenerAdded() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   774
        flags |= IS_LISTENER_ADDED;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   775
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   776
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   777
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   778
     * Return only the flags that should be copied during cloning.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   779
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   780
     * @return Subset of flags that should be copied.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   781
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   782
    private int getClonedFlags() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   783
        return flags & CLONEABLE_FLAGS_MASK;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   784
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   785
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   786
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   787
     * {@link PropertyMap} iterator.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   788
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   789
    private static class PropertyMapIterator implements Iterator<Object> {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   790
        /** Property iterator. */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   791
        final Iterator<Property> iter;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   792
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   793
        /** Current Property. */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   794
        Property property;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   795
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   796
        /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   797
         * Constructor.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   798
         *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   799
         * @param propertyMap {@link PropertyMap} to iterate over.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   800
         */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   801
        PropertyMapIterator(final PropertyMap propertyMap) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   802
            iter = Arrays.asList(propertyMap.properties.getProperties()).iterator();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   803
            property = iter.hasNext() ? iter.next() : null;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   804
            skipNotEnumerable();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   805
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   806
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   807
        /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   808
         * Ignore properties that are not enumerable.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   809
         */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   810
        private void skipNotEnumerable() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   811
            while (property != null && !property.isEnumerable()) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   812
                property = iter.hasNext() ? iter.next() : null;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   813
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   814
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   815
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   816
        @Override
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   817
        public boolean hasNext() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   818
            return property != null;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   819
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   820
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   821
        @Override
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   822
        public Object next() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   823
            if (property == null) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   824
                throw new NoSuchElementException();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   825
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   826
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   827
            final Object key = property.getKey();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   828
            property = iter.next();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   829
            skipNotEnumerable();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   830
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   831
            return key;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   832
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   833
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   834
        @Override
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   835
        public void remove() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   836
            throw new UnsupportedOperationException();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   837
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   838
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   839
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   840
    /*
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   841
     * PropertyListener implementation.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   842
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   843
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   844
    @Override
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   845
    public void propertyAdded(final ScriptObject object, final Property prop) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   846
        invalidateProtoGetSwitchPoint(prop);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   847
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   848
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   849
    @Override
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   850
    public void propertyDeleted(final ScriptObject object, final Property prop) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   851
        invalidateProtoGetSwitchPoint(prop);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   852
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   853
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   854
    @Override
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   855
    public void propertyModified(final ScriptObject object, final Property oldProp, final Property newProp) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   856
        invalidateProtoGetSwitchPoint(oldProp);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   857
    }
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
     * Debugging and statistics.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   861
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   862
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   863
    // counters updated only in debug mode
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   864
    private static int count;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   865
    private static int clonedCount;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   866
    private static int historyHit;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   867
    private static int protoInvalidations;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   868
    private static int protoHistoryHit;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   869
    private static int setProtoNewMapCount;
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
     * @return Total number of maps.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   873
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   874
    public static int getCount() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   875
        return count;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   876
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   877
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   878
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   879
     * @return The number of maps that were cloned.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   880
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   881
    public static int getClonedCount() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   882
        return clonedCount;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   883
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   884
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   885
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   886
     * @return The number of times history was successfully used.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   887
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   888
    public static int getHistoryHit() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   889
        return historyHit;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   890
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   891
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   892
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   893
     * @return The number of times prototype changes caused invalidation.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   894
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   895
    public static int getProtoInvalidations() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   896
        return protoInvalidations;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   897
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   898
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   899
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   900
     * @return The number of times proto history was successfully used.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   901
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   902
    public static int getProtoHistoryHit() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   903
        return protoHistoryHit;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   904
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   905
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   906
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   907
     * @return The number of times prototypes were modified.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   908
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   909
    public static int getSetProtoNewMapCount() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   910
        return setProtoNewMapCount;
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
     * Increment the prototype set count.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   915
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   916
    private static void incrementSetProtoNewMapCount() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   917
        setProtoNewMapCount++;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   918
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   919
}
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   920