From ab18b431cb462044537f361ab0ef875c0b8e76a7 Mon Sep 17 00:00:00 2001 From: James Laskey Date: Tue, 30 Apr 2013 10:05:42 -0300 Subject: [PATCH 01/15] 8006220: Simplify PropertyMaps Reviewed-by: hannesw, lagergren --- .../nashorn/internal/codegen/MapCreator.java | 6 +- .../codegen/ObjectClassGenerator.java | 10 + .../internal/codegen/ObjectCreator.java | 13 +- .../nashorn/internal/objects/NativeDebug.java | 60 ---- .../internal/objects/NativeJSAdapter.java | 6 +- .../internal/runtime/AccessorProperty.java | 15 +- .../jdk/nashorn/internal/runtime/Context.java | 3 - .../nashorn/internal/runtime/Property.java | 23 +- .../internal/runtime/PropertyHashMap.java | 4 +- .../nashorn/internal/runtime/PropertyMap.java | 194 ++++++------ .../internal/runtime/ScriptObject.java | 286 +++++++----------- .../internal/runtime/SetMethodCreator.java | 28 +- .../internal/runtime/SpillProperty.java | 85 ------ .../internal/runtime/StructureLoader.java | 3 +- .../runtime/UserAccessorProperty.java | 9 +- .../src/jdk/nashorn/internal/scripts/JO.java | 3 +- nashorn/src/jdk/nashorn/tools/Shell.java | 2 +- 17 files changed, 267 insertions(+), 483 deletions(-) delete mode 100644 nashorn/src/jdk/nashorn/internal/runtime/SpillProperty.java diff --git a/nashorn/src/jdk/nashorn/internal/codegen/MapCreator.java b/nashorn/src/jdk/nashorn/internal/codegen/MapCreator.java index 00f3f80f55b..153e0367f3f 100644 --- a/nashorn/src/jdk/nashorn/internal/codegen/MapCreator.java +++ b/nashorn/src/jdk/nashorn/internal/codegen/MapCreator.java @@ -65,10 +65,12 @@ public class MapCreator { * Constructs a property map based on a set of fields. * * @param hasArguments does the created object have an "arguments" property + * @param fieldCount Number of fields in use. + * @param fieldMaximum Number of fields available. * * @return New map populated with accessor properties. */ - PropertyMap makeMap(final boolean hasArguments) { + PropertyMap makeMap(final boolean hasArguments, final int fieldCount, final int fieldMaximum) { final List properties = new ArrayList<>(); assert keys != null; @@ -82,7 +84,7 @@ public class MapCreator { } } - return PropertyMap.newMap(structure, properties); + return PropertyMap.newMap(structure, properties, fieldCount, fieldMaximum); } /** diff --git a/nashorn/src/jdk/nashorn/internal/codegen/ObjectClassGenerator.java b/nashorn/src/jdk/nashorn/internal/codegen/ObjectClassGenerator.java index b75145262b7..172c1035426 100644 --- a/nashorn/src/jdk/nashorn/internal/codegen/ObjectClassGenerator.java +++ b/nashorn/src/jdk/nashorn/internal/codegen/ObjectClassGenerator.java @@ -68,6 +68,16 @@ public final class ObjectClassGenerator { */ static final String SCOPE_MARKER = "P"; + /** + * Minimum number of extra fields in an object. + */ + static final int FIELD_PADDING = 4; + + /** + * Rounding when calculating the number of fields. + */ + static final int FIELD_ROUNDING = 4; + /** * Debug field logger * Should we print debugging information for fields when they are generated and getters/setters are called? diff --git a/nashorn/src/jdk/nashorn/internal/codegen/ObjectCreator.java b/nashorn/src/jdk/nashorn/internal/codegen/ObjectCreator.java index a9c494ccab0..82efd14c4fa 100644 --- a/nashorn/src/jdk/nashorn/internal/codegen/ObjectCreator.java +++ b/nashorn/src/jdk/nashorn/internal/codegen/ObjectCreator.java @@ -26,6 +26,8 @@ package jdk.nashorn.internal.codegen; import java.util.List; +import static jdk.nashorn.internal.codegen.ObjectClassGenerator.FIELD_PADDING; +import static jdk.nashorn.internal.codegen.ObjectClassGenerator.FIELD_ROUNDING; import jdk.nashorn.internal.ir.Symbol; import jdk.nashorn.internal.runtime.Context; import jdk.nashorn.internal.runtime.PropertyMap; @@ -50,6 +52,7 @@ public abstract class ObjectCreator { private final boolean isScope; private final boolean hasArguments; private int fieldCount; + private int paddedFieldCount; private int paramCount; private String fieldObjectClassName; private Class fieldObjectClass; @@ -88,6 +91,8 @@ public abstract class ObjectCreator { } } } + + paddedFieldCount = (fieldCount + FIELD_PADDING + FIELD_ROUNDING - 1) / FIELD_ROUNDING * FIELD_ROUNDING; } /** @@ -96,7 +101,7 @@ public abstract class ObjectCreator { private void findClass() { fieldObjectClassName = isScope() ? ObjectClassGenerator.getClassName(fieldCount, paramCount) : - ObjectClassGenerator.getClassName(fieldCount); + ObjectClassGenerator.getClassName(paddedFieldCount); try { this.fieldObjectClass = Context.forStructureClass(Compiler.binaryName(fieldObjectClassName)); @@ -125,11 +130,7 @@ public abstract class ObjectCreator { * @return the newly created property map */ protected PropertyMap makeMap() { - if (keys.isEmpty()) { //empty map - propertyMap = PropertyMap.newMap(fieldObjectClass); - } else { - propertyMap = newMapCreator(fieldObjectClass).makeMap(hasArguments()); - } + propertyMap = newMapCreator(fieldObjectClass).makeMap(hasArguments(), fieldCount, paddedFieldCount); return propertyMap; } diff --git a/nashorn/src/jdk/nashorn/internal/objects/NativeDebug.java b/nashorn/src/jdk/nashorn/internal/objects/NativeDebug.java index aba76c8e9fe..337f2e3d5b1 100644 --- a/nashorn/src/jdk/nashorn/internal/objects/NativeDebug.java +++ b/nashorn/src/jdk/nashorn/internal/objects/NativeDebug.java @@ -86,66 +86,6 @@ public final class NativeDebug extends ScriptObject { return UNDEFINED; } - /** - * Nashorn extension: get embed0 from {@link ScriptObject} - * - * @param self self reference - * @param obj script object - * @return the embed0 property value for the given ScriptObject - */ - @Function(attributes = Attribute.NOT_ENUMERABLE, where = Where.CONSTRUCTOR) - public static Object embed0(final Object self, final Object obj) { - if (obj instanceof ScriptObject) { - return ((ScriptObject)obj).embed0; - } - return UNDEFINED; - } - - /** - * Nashorn extension: get embed1 from {@link ScriptObject} - * - * @param self self reference - * @param obj script object - * @return the embed1 property value for the given ScriptObject - */ - @Function(attributes = Attribute.NOT_ENUMERABLE, where = Where.CONSTRUCTOR) - public static Object embed1(final Object self, final Object obj) { - if (obj instanceof ScriptObject) { - return ((ScriptObject)obj).embed1; - } - return UNDEFINED; - } - - /** - * Nashorn extension: get embed2 from {@link ScriptObject} - * - * @param self self reference - * @param obj script object - * @return the embed2 property value for the given ScriptObject - */ - @Function(attributes = Attribute.NOT_ENUMERABLE, where = Where.CONSTRUCTOR) - public static Object embed2(final Object self, final Object obj) { - if (obj instanceof ScriptObject) { - return ((ScriptObject)obj).embed2; - } - return UNDEFINED; - } - - /** - * Nashorn extension: get embed3 from {@link ScriptObject} - * - * @param self self reference - * @param obj script object - * @return the embed3 property value for the given ScriptObject - */ - @Function(attributes = Attribute.NOT_ENUMERABLE, where = Where.CONSTRUCTOR) - public static Object embed3(final Object self, final Object obj) { - if (obj instanceof ScriptObject) { - return ((ScriptObject)obj).embed3; - } - return UNDEFINED; - } - /** * Nashorn extension: get spill vector from {@link ScriptObject} * diff --git a/nashorn/src/jdk/nashorn/internal/objects/NativeJSAdapter.java b/nashorn/src/jdk/nashorn/internal/objects/NativeJSAdapter.java index 8a3f42d2c1c..153db769f9b 100644 --- a/nashorn/src/jdk/nashorn/internal/objects/NativeJSAdapter.java +++ b/nashorn/src/jdk/nashorn/internal/objects/NativeJSAdapter.java @@ -620,7 +620,7 @@ public final class NativeJSAdapter extends ScriptObject { // to name. Probably not a big deal, but if we can ever make it leaner, it'd be nice. return new GuardedInvocation(MH.dropArguments(MH.constant(Object.class, func.makeBoundFunction(this, new Object[] { name })), 0, Object.class), - adaptee.getMap().getProtoGetSwitchPoint(__call__), testJSAdaptor(adaptee, null, null, null)); + adaptee.getMap().getProtoGetSwitchPoint(adaptee.getProto(), __call__), testJSAdaptor(adaptee, null, null, null)); } throw typeError("no.such.function", desc.getNameToken(2), ScriptRuntime.safeToString(this)); default: @@ -687,7 +687,7 @@ public final class NativeJSAdapter extends ScriptObject { if (methodHandle != null) { return new GuardedInvocation( methodHandle, - adaptee.getMap().getProtoGetSwitchPoint(hook), + adaptee.getMap().getProtoGetSwitchPoint(adaptee.getProto(), hook), testJSAdaptor(adaptee, findData.getGetter(Object.class), findData.getOwner(), func)); } } @@ -699,7 +699,7 @@ public final class NativeJSAdapter extends ScriptObject { final MethodHandle methodHandle = hook.equals(__put__) ? MH.asType(Lookup.EMPTY_SETTER, type) : Lookup.emptyGetter(type.returnType()); - return new GuardedInvocation(methodHandle, adaptee.getMap().getProtoGetSwitchPoint(hook), testJSAdaptor(adaptee, null, null, null)); + return new GuardedInvocation(methodHandle, adaptee.getMap().getProtoGetSwitchPoint(adaptee.getProto(), hook), testJSAdaptor(adaptee, null, null, null)); } } diff --git a/nashorn/src/jdk/nashorn/internal/runtime/AccessorProperty.java b/nashorn/src/jdk/nashorn/internal/runtime/AccessorProperty.java index b0ca46f78f9..38effdafd55 100644 --- a/nashorn/src/jdk/nashorn/internal/runtime/AccessorProperty.java +++ b/nashorn/src/jdk/nashorn/internal/runtime/AccessorProperty.java @@ -50,8 +50,6 @@ import jdk.nashorn.internal.lookup.MethodHandleFactory; /** * An AccessorProperty is the most generic property type. An AccessorProperty is * represented as fields in a ScriptObject class. - * - * @see SpillProperty */ public class AccessorProperty extends Property { private static final MethodHandles.Lookup lookup = MethodHandles.lookup(); @@ -77,6 +75,7 @@ public class AccessorProperty extends Property { private static final MethodType[] ACCESSOR_GETTER_TYPES = new MethodType[NOOF_TYPES]; private static final MethodType[] ACCESSOR_SETTER_TYPES = new MethodType[NOOF_TYPES]; + private static final MethodHandle SPILLGETTER = MH.asType(MH.getter(MethodHandles.lookup(), ScriptObject.class, "spill", Object[].class), Lookup.GET_OBJECT_TYPE); /** Seed getter for the primitive version of this field (in -Dnashorn.fields.dual=true mode) */ private MethodHandle primitiveGetter; @@ -285,7 +284,7 @@ public class AccessorProperty extends Property { "get"); } - return getters[i]; + return isSpill() ? MH.filterArguments(getters[i], 0, SPILLGETTER) : getters[i]; } private Property getWiderProperty(final Class type) { @@ -327,6 +326,7 @@ public class AccessorProperty extends Property { final Class forType = currentType == null ? type : currentType; //if we are asking for an object setter, but are still a primitive type, we might try to box it + MethodHandle mh; if (needsInvalidator(i, ci)) { final Property newProperty = getWiderProperty(type); @@ -335,12 +335,15 @@ public class AccessorProperty extends Property { final MethodHandle explodeTypeSetter = MH.filterArguments(widerSetter, 0, MH.insertArguments(REPLACE_MAP, 1, newMap, getKey(), currentType, type)); if (currentType != null && currentType.isPrimitive() && type == Object.class) { //might try a box check on this to avoid widening field to object storage - return createGuardBoxedPrimitiveSetter(currentType, generateSetter(currentType, currentType), explodeTypeSetter); + mh = createGuardBoxedPrimitiveSetter(currentType, generateSetter(currentType, currentType), explodeTypeSetter); + } else { + mh = explodeTypeSetter; } - return explodeTypeSetter; + } else { + mh = generateSetter(forType, type); } - return generateSetter(forType, type); + return isSpill() ? MH.filterArguments(mh, 0, SPILLGETTER) : mh; } @Override diff --git a/nashorn/src/jdk/nashorn/internal/runtime/Context.java b/nashorn/src/jdk/nashorn/internal/runtime/Context.java index 40fe1ba15df..c3d32d0ba72 100644 --- a/nashorn/src/jdk/nashorn/internal/runtime/Context.java +++ b/nashorn/src/jdk/nashorn/internal/runtime/Context.java @@ -201,9 +201,6 @@ public final class Context { /** Current error manager. */ private final ErrorManager errors; - /** Empty map used for seed map for JO objects */ - final PropertyMap emptyMap = PropertyMap.newEmptyMap(this); - private static final ClassLoader myLoader = Context.class.getClassLoader(); private static final StructureLoader sharedLoader; diff --git a/nashorn/src/jdk/nashorn/internal/runtime/Property.java b/nashorn/src/jdk/nashorn/internal/runtime/Property.java index dfb8bf5d642..585fc4b3113 100644 --- a/nashorn/src/jdk/nashorn/internal/runtime/Property.java +++ b/nashorn/src/jdk/nashorn/internal/runtime/Property.java @@ -41,7 +41,6 @@ import jdk.nashorn.internal.codegen.types.Type; * * @see PropertyMap * @see AccessorProperty - * @see SpillProperty * @see UserAccessorProperty */ public abstract class Property { @@ -64,7 +63,7 @@ public abstract class Property { private static final int MODIFY_MASK = 0b0000_0000_1111; - /** Is this a spill property? See {@link SpillProperty} */ + /** Is this a spill property? See {@link AccessorProperty} */ public static final int IS_SPILL = 0b0000_0001_0000; /** Is this a function parameter? */ @@ -88,7 +87,7 @@ public abstract class Property { /** Property flags. */ protected int flags; - /** Property field number or spill slot */ + /** Property field number or spill slot. */ private final int slot; /** @@ -248,7 +247,7 @@ public abstract class Property { * Does this property use any slots in the spill array described in * {@link Property#isSpill}? In that case how many. Currently a property * only uses max one spill slot, but this may change in future representations - * Only {@link SpillProperty} instances use spill slots + * Only {@link AccessorProperty} instances use spill slots * * @return number of spill slots a property is using */ @@ -344,6 +343,14 @@ public abstract class Property { return key; } + /** + * Get the field number or spill slot + * @return number/slot, -1 if none exists + */ + public int getSlot() { + return slot; + } + /** * Abstract method for retrieving the setter for the property. We do not know * anything about the internal representation when we request the setter, we only @@ -388,14 +395,6 @@ public abstract class Property { return null; } - /** - * Get the field number or spill slot - * @return number/slot, -1 if none exists - */ - public int getSlot() { - return slot; - } - @Override public int hashCode() { final Class type = getCurrentType(); diff --git a/nashorn/src/jdk/nashorn/internal/runtime/PropertyHashMap.java b/nashorn/src/jdk/nashorn/internal/runtime/PropertyHashMap.java index 6e41fd56827..bfa06920ae3 100644 --- a/nashorn/src/jdk/nashorn/internal/runtime/PropertyHashMap.java +++ b/nashorn/src/jdk/nashorn/internal/runtime/PropertyHashMap.java @@ -110,7 +110,7 @@ public final class PropertyHashMap implements Map { private static final int LIST_THRESHOLD = 8; /** Initial map. */ - public static final PropertyHashMap EMPTY_MAP = new PropertyHashMap(); + public static final PropertyHashMap EMPTY_HASHMAP = new PropertyHashMap(); /** Number of properties in the map. */ private final int size; @@ -246,7 +246,7 @@ public final class PropertyHashMap implements Map { } } else if (findElement(list, key) != null) { final int newSize = size - 1; - return newSize != 0 ? new PropertyHashMap(newSize, null, removeFromList(list, key)) : EMPTY_MAP; + return newSize != 0 ? new PropertyHashMap(newSize, null, removeFromList(list, key)) : EMPTY_HASHMAP; } return this; } diff --git a/nashorn/src/jdk/nashorn/internal/runtime/PropertyMap.java b/nashorn/src/jdk/nashorn/internal/runtime/PropertyMap.java index b5154c853f6..c01ceae0d18 100644 --- a/nashorn/src/jdk/nashorn/internal/runtime/PropertyMap.java +++ b/nashorn/src/jdk/nashorn/internal/runtime/PropertyMap.java @@ -25,7 +25,7 @@ package jdk.nashorn.internal.runtime; -import static jdk.nashorn.internal.runtime.PropertyHashMap.EMPTY_MAP; +import static jdk.nashorn.internal.runtime.PropertyHashMap.EMPTY_HASHMAP; import java.lang.invoke.MethodHandle; import java.lang.invoke.SwitchPoint; @@ -49,29 +49,27 @@ import java.util.WeakHashMap; * will return a new map. */ public final class PropertyMap implements Iterable, PropertyListener { - /** Is this a prototype PropertyMap? */ - public static final int IS_PROTOTYPE = 0b0000_0001; /** Used for non extensible PropertyMaps, negative logic as the normal case is extensible. See {@link ScriptObject#preventExtensions()} */ - public static final int NOT_EXTENSIBLE = 0b0000_0010; + public static final int NOT_EXTENSIBLE = 0b0000_0001; /** This mask is used to preserve certain flags when cloning the PropertyMap. Others should not be copied */ private static final int CLONEABLE_FLAGS_MASK = 0b0000_1111; /** Has a listener been added to this property map. This flag is not copied when cloning a map. See {@link PropertyListener} */ public static final int IS_LISTENER_ADDED = 0b0001_0000; + /** Empty map used for seed map for JO$ objects */ + private static final PropertyMap EMPTY_MAP = new PropertyMap(EMPTY_HASHMAP); + /** Map status flags. */ private int flags; - /** Class of object referenced.*/ - private final Class structure; - - /** Context associated with this {@link PropertyMap}. */ - private final Context context; - /** Map of properties. */ private final PropertyHashMap properties; - /** objects proto. */ - private ScriptObject proto; + /** Number of fields in use. */ + private int fieldCount; + + /** Number of fields available. */ + private int fieldMaximum; /** Length of spill in use. */ private int spillLength; @@ -91,21 +89,30 @@ public final class PropertyMap implements Iterable, PropertyListener { /** * Constructor. * - * @param structure Class the map's {@link AccessorProperty}s apply to. - * @param context Context associated with this {@link PropertyMap}. - * @param properties A {@link PropertyHashMap} with initial contents. + * @param properties A {@link PropertyHashMap} with initial contents. + * @param fieldCount Number of fields in use. + * @param fieldMaximum Number of fields available. */ - PropertyMap(final Class structure, final Context context, final PropertyHashMap properties) { - this.structure = structure; - this.context = context; - this.properties = properties; - this.hashCode = computeHashCode(); + private PropertyMap(final PropertyHashMap properties, final int fieldCount, final int fieldMaximum) { + this.properties = properties; + this.hashCode = computeHashCode(); + this.fieldCount = fieldCount; + this.fieldMaximum = fieldMaximum; if (Context.DEBUG) { count++; } } + /** + * Constructor. + * + * @param properties A {@link PropertyHashMap} with initial contents. + */ + private PropertyMap(final PropertyHashMap properties) { + this(properties, 0, 0); + } + /** * Cloning constructor. * @@ -113,13 +120,12 @@ public final class PropertyMap implements Iterable, PropertyListener { * @param properties A {@link PropertyHashMap} with a new set of properties. */ private PropertyMap(final PropertyMap propertyMap, final PropertyHashMap properties) { - this.structure = propertyMap.structure; - this.context = propertyMap.context; - this.properties = properties; - this.flags = propertyMap.getClonedFlags(); - this.proto = propertyMap.proto; - this.spillLength = propertyMap.spillLength; - this.hashCode = computeHashCode(); + this.properties = properties; + this.flags = propertyMap.getClonedFlags(); + this.spillLength = propertyMap.spillLength; + this.fieldCount = propertyMap.fieldCount; + this.fieldMaximum = propertyMap.fieldMaximum; + this.hashCode = computeHashCode(); if (Context.DEBUG) { count++; @@ -127,6 +133,15 @@ public final class PropertyMap implements Iterable, PropertyListener { } } + /** + * Cloning constructor. + * + * @param propertyMap Existing property map. + */ + private PropertyMap(final PropertyMap propertyMap) { + this(propertyMap, propertyMap.properties); + } + /** * Duplicates this PropertyMap instance. This is used by nasgen generated * prototype and constructor classes. {@link PropertyMap} used for singletons @@ -138,7 +153,7 @@ public final class PropertyMap implements Iterable, PropertyListener { * @return Duplicated {@link PropertyMap}. */ public PropertyMap duplicate() { - return new PropertyMap(this.structure, this.context, this.properties); + return new PropertyMap(this.properties); } /** @@ -146,20 +161,20 @@ public final class PropertyMap implements Iterable, PropertyListener { * * @param structure Class the map's {@link AccessorProperty}s apply to. * @param properties Collection of initial properties. + * @param fieldCount Number of fields in use. + * @param fieldMaximum Number of fields available. * * @return New {@link PropertyMap}. */ - public static PropertyMap newMap(final Class structure, final Collection properties) { - final Context context = Context.fromClass(structure); - + public static PropertyMap newMap(final Class structure, final Collection properties, final int fieldCount, final int fieldMaximum) { // Reduce the number of empty maps in the context. if (structure == jdk.nashorn.internal.scripts.JO.class) { - return context.emptyMap; + return EMPTY_MAP; } - PropertyHashMap newProperties = EMPTY_MAP.immutableAdd(properties); + PropertyHashMap newProperties = EMPTY_HASHMAP.immutableAdd(properties); - return new PropertyMap(structure, context, newProperties); + return new PropertyMap(newProperties, fieldCount, fieldMaximum); } /** @@ -170,7 +185,7 @@ public final class PropertyMap implements Iterable, PropertyListener { * @return New {@link PropertyMap}. */ public static PropertyMap newMap(final Class structure) { - return newMap(structure, null); + return newMap(structure, null, 0, 0); } /** @@ -180,7 +195,7 @@ public final class PropertyMap implements Iterable, PropertyListener { * @return New empty {@link PropertyMap}. */ public static PropertyMap newEmptyMap(final Context context) { - return new PropertyMap(jdk.nashorn.internal.scripts.JO.class, context, EMPTY_MAP); + return new PropertyMap(EMPTY_HASHMAP); } /** @@ -195,11 +210,12 @@ public final class PropertyMap implements Iterable, PropertyListener { /** * Return a SwitchPoint used to track changes of a property in a prototype. * - * @param key {@link Property} key. + * @param proto Object prototype. + * @param key {@link Property} key. * * @return A shared {@link SwitchPoint} for the property. */ - public SwitchPoint getProtoGetSwitchPoint(final String key) { + public SwitchPoint getProtoGetSwitchPoint(final ScriptObject proto, final String key) { if (proto == null) { return null; } @@ -295,6 +311,11 @@ public final class PropertyMap implements Iterable, PropertyListener { final PropertyHashMap newProperties = properties.immutableAdd(property); newMap = new PropertyMap(this, newProperties); addToHistory(property, newMap); + + if(!property.isSpill()) { + newMap.fieldCount = Math.max(newMap.fieldCount, property.getSlot() + 1); + } + newMap.spillLength += property.getSpillCount(); } @@ -355,7 +376,6 @@ public final class PropertyMap implements Iterable, PropertyListener { newProperty instanceof UserAccessorProperty) : "arbitrary replaceProperty attempted"; newMap.flags = getClonedFlags(); - newMap.proto = proto; /* * spillLength remains same in case (1) and (2) because of slot reuse. Only for case (3), we need @@ -411,7 +431,7 @@ public final class PropertyMap implements Iterable, PropertyListener { * @return New map with {@link #NOT_EXTENSIBLE} flag set. */ PropertyMap preventExtensions() { - final PropertyMap newMap = new PropertyMap(this, this.properties); + final PropertyMap newMap = new PropertyMap(this); newMap.flags |= NOT_EXTENSIBLE; return newMap; } @@ -423,7 +443,7 @@ public final class PropertyMap implements Iterable, PropertyListener { * {@link Property#NOT_CONFIGURABLE} set. */ PropertyMap seal() { - PropertyHashMap newProperties = EMPTY_MAP; + PropertyHashMap newProperties = EMPTY_HASHMAP; for (final Property oldProperty : properties.getProperties()) { newProperties = newProperties.immutableAdd(oldProperty.addFlags(Property.NOT_CONFIGURABLE)); @@ -442,7 +462,7 @@ public final class PropertyMap implements Iterable, PropertyListener { * {@link Property#NOT_CONFIGURABLE} and {@link Property#NOT_WRITABLE} set. */ PropertyMap freeze() { - PropertyHashMap newProperties = EMPTY_MAP; + PropertyHashMap newProperties = EMPTY_HASHMAP; for (Property oldProperty : properties.getProperties()) { int propertyFlags = Property.NOT_CONFIGURABLE; @@ -578,11 +598,7 @@ public final class PropertyMap implements Iterable, PropertyListener { * @return Computed hash code. */ private int computeHashCode() { - int hash = structure.hashCode(); - - if (proto != null) { - hash ^= proto.hashCode(); - } + int hash = 0; for (final Property property : getProperties()) { hash = hash << 7 ^ hash >> 7; @@ -605,9 +621,7 @@ public final class PropertyMap implements Iterable, PropertyListener { final PropertyMap otherMap = (PropertyMap)other; - if (structure != otherMap.structure || - proto != otherMap.proto || - properties.size() != otherMap.properties.size()) { + if (properties.size() != otherMap.properties.size()) { return false; } @@ -658,31 +672,6 @@ public final class PropertyMap implements Iterable, PropertyListener { return new PropertyMapIterator(this); } - /** - * Return map's {@link Context}. - * - * @return The {@link Context} where the map originated. - */ - Context getContext() { - return context; - } - - /** - * Check if this map is a prototype - * - * @return {@code true} if is prototype - */ - public boolean isPrototype() { - return (flags & IS_PROTOTYPE) != 0; - } - - /** - * Flag this map as having a prototype. - */ - private void setIsPrototype() { - flags |= IS_PROTOTYPE; - } - /** * Check whether a {@link PropertyListener} has been added to this map. * @@ -720,6 +709,22 @@ public final class PropertyMap implements Iterable, PropertyListener { boolean isFrozen() { return !isExtensible() && allFrozen(); } + /** + * Get the number of fields allocated for this {@link PropertyMap}. + * + * @return Number of fields allocated. + */ + int getFieldCount() { + return fieldCount; + } + /** + * Get maximum number of fields available for this {@link PropertyMap}. + * + * @return Number of fields available. + */ + int getFieldMaximum() { + return fieldMaximum; + } /** * Get length of spill area associated with this {@link PropertyMap}. @@ -731,25 +736,20 @@ public final class PropertyMap implements Iterable, PropertyListener { } /** - * Return the prototype of objects associated with this {@link PropertyMap}. + * Change the prototype of objects associated with this {@link PropertyMap}. * - * @return Prototype object. - */ - ScriptObject getProto() { - return proto; - } - - /** - * Set the prototype of objects associated with this {@link PropertyMap}. - * - * @param newProto Prototype object to use. + * @param oldProto Current prototype object. + * @param newProto New prototype object to replace oldProto. * * @return New {@link PropertyMap} with prototype changed. */ - PropertyMap setProto(final ScriptObject newProto) { - final ScriptObject oldProto = this.proto; - - if (oldProto == newProto) { + PropertyMap changeProto(final ScriptObject oldProto, final ScriptObject newProto) { + if ((oldProto == newProto) || + (size() == 0 && + oldProto == null && + protoGetSwitches == null && + history == null && + protoHistory == null)) { return this; } @@ -761,19 +761,10 @@ public final class PropertyMap implements Iterable, PropertyListener { if (Context.DEBUG) { incrementSetProtoNewMapCount(); } - final PropertyMap newMap = new PropertyMap(this, this.properties); + + final PropertyMap newMap = new PropertyMap(this); addToProtoHistory(newProto, newMap); - newMap.proto = newProto; - - if (oldProto != null && newMap.isListenerAdded()) { - oldProto.removePropertyListener(newMap); - } - - if (newProto != null) { - newProto.getMap().setIsPrototype(); - } - return newMap; } @@ -927,4 +918,3 @@ public final class PropertyMap implements Iterable, PropertyListener { setProtoNewMapCount++; } } - diff --git a/nashorn/src/jdk/nashorn/internal/runtime/ScriptObject.java b/nashorn/src/jdk/nashorn/internal/runtime/ScriptObject.java index 935172016b8..18724fb1cad 100644 --- a/nashorn/src/jdk/nashorn/internal/runtime/ScriptObject.java +++ b/nashorn/src/jdk/nashorn/internal/runtime/ScriptObject.java @@ -104,34 +104,31 @@ public abstract class ScriptObject extends PropertyListenerManager implements Pr /** Per ScriptObject flag - is this an arguments object? */ public static final int IS_ARGUMENTS = 0b0000_0100; + /** Is this a prototype PropertyMap? */ + public static final int IS_PROTOTYPE = 0b0000_1000; + /** Spill growth rate - by how many elements does {@link ScriptObject#spill} when full */ public static final int SPILL_RATE = 8; /** Map to property information and accessor functions. Ordered by insertion. */ private PropertyMap map; + /** objects proto. */ + private ScriptObject proto; + + /** Context of the object, lazily cached. */ + private Context context; + /** Object flags. */ private int flags; - /** Area for properties added to object after instantiation, see {@link SpillProperty} */ + /** Area for properties added to object after instantiation, see {@link AccessorProperty} */ public Object[] spill; - /** Local embed area position 0 - used for {@link SpillProperty} before {@link ScriptObject#spill} */ - public Object embed0; - - /** Local embed area position 1 - used for {@link SpillProperty} before {@link ScriptObject#spill} */ - public Object embed1; - - /** Local embed area position 2 - used for {@link SpillProperty} before {@link ScriptObject#spill} */ - public Object embed2; - - /** Local embed area position 3 - used for {@link SpillProperty} before {@link ScriptObject#spill} */ - public Object embed3; - /** Indexed array data. */ private ArrayData arrayData; - static final MethodHandle SETEMBED = findOwnMH("setEmbed", void.class, CallSiteDescriptor.class, PropertyMap.class, PropertyMap.class, MethodHandle.class, int.class, Object.class, Object.class); + static final MethodHandle SETFIELD = findOwnMH("setField", void.class, CallSiteDescriptor.class, PropertyMap.class, PropertyMap.class, MethodHandle.class, Object.class, Object.class); static final MethodHandle SETSPILL = findOwnMH("setSpill", void.class, CallSiteDescriptor.class, PropertyMap.class, PropertyMap.class, int.class, Object.class, Object.class); static final MethodHandle SETSPILLWITHNEW = findOwnMH("setSpillWithNew", void.class, CallSiteDescriptor.class, PropertyMap.class, PropertyMap.class, int.class, Object.class, Object.class); static final MethodHandle SETSPILLWITHGROW = findOwnMH("setSpillWithGrow", void.class, CallSiteDescriptor.class, PropertyMap.class, PropertyMap.class, int.class, int.class, Object.class, Object.class); @@ -783,8 +780,8 @@ public abstract class ScriptObject extends PropertyListenerManager implements Pr // delete getter and setter function references so that we don't leak if (property instanceof UserAccessorProperty) { final UserAccessorProperty uc = (UserAccessorProperty) property; - setEmbedOrSpill(uc.getGetterSlot(), null); - setEmbedOrSpill(uc.getSetterSlot(), null); + setSpill(uc.getGetterSlot(), null); + setSpill(uc.getSetterSlot(), null); } return true; } @@ -809,7 +806,7 @@ public abstract class ScriptObject extends PropertyListenerManager implements Pr int getterSlot = uc.getGetterSlot(); // clear the old getter and set the new getter - setEmbedOrSpill(getterSlot, getter); + setSpill(getterSlot, getter); // if getter function is null, flag the slot to be negative (less by 1) if (getter == null) { getterSlot = -getterSlot - 1; @@ -817,7 +814,7 @@ public abstract class ScriptObject extends PropertyListenerManager implements Pr int setterSlot = uc.getSetterSlot(); // clear the old setter and set the new setter - setEmbedOrSpill(setterSlot, setter); + setSpill(setterSlot, setter); // if setter function is null, flag the slot to be negative (less by 1) if (setter == null) { setterSlot = -setterSlot - 1; @@ -1056,8 +1053,11 @@ public abstract class ScriptObject extends PropertyListenerManager implements Pr * Return the current context from the object's map. * @return Current context. */ - final Context getContext() { - return getMap().getContext(); + protected final Context getContext() { + if (context == null) { + context = Context.fromClass(getClass()); + } + return context; } /** @@ -1097,44 +1097,30 @@ public abstract class ScriptObject extends PropertyListenerManager implements Pr * @return __proto__ object. */ public final ScriptObject getProto() { - return getMap().getProto(); - } - - /** - * Check if this is a prototype - * @return true if {@link PropertyMap#isPrototype()} is true for this ScriptObject - */ - public final boolean isPrototype() { - return getMap().isPrototype(); + return proto; } /** * Set the __proto__ of an object. * @param newProto new __proto__ to set. */ - public final void setProto(final ScriptObject newProto) { - PropertyMap oldMap = getMap(); - ScriptObject oldProto = getProto(); + public synchronized final void setProto(final ScriptObject newProto) { + final ScriptObject oldProto = proto; + map = map.changeProto(oldProto, newProto); - while (oldProto != newProto) { - final PropertyMap newMap = oldMap.setProto(newProto); + if (newProto != null) { + newProto.setIsPrototype(); + } - if (!compareAndSetMap(oldMap, newMap)) { - oldMap = getMap(); - oldProto = getProto(); - } else { - if (isPrototype()) { + proto = newProto; - if (oldProto != null) { - oldProto.removePropertyListener(this); - } + if (isPrototype()) { + if (oldProto != null) { + oldProto.removePropertyListener(this); + } - if (newProto != null) { - newProto.addPropertyListener(this); - } - } - - return; + if (newProto != null) { + newProto.addPropertyListener(this); } } } @@ -1326,6 +1312,22 @@ public abstract class ScriptObject extends PropertyListenerManager implements Pr flags |= IS_ARGUMENTS; } + /** + * Check if this object is a prototype + * + * @return {@code true} if is prototype + */ + public boolean isPrototype() { + return (flags & IS_PROTOTYPE) != 0; + } + + /** + * Flag this object as having a prototype. + */ + public void setIsPrototype() { + flags |= IS_PROTOTYPE; + } + /** * Get the {@link ArrayData} for this ScriptObject if it is an array * @return array data @@ -1719,11 +1721,11 @@ public abstract class ScriptObject extends PropertyListenerManager implements Pr if (!property.hasGetterFunction()) { methodHandle = bindTo(methodHandle, prototype); } - return new GuardedInvocation(methodHandle, getMap().getProtoGetSwitchPoint(name), guard); + return new GuardedInvocation(methodHandle, getMap().getProtoGetSwitchPoint(proto, name), guard); } assert !NashornCallSiteDescriptor.isFastScope(desc); - return new GuardedInvocation(Lookup.emptyGetter(returnType), getMap().getProtoGetSwitchPoint(name), guard); + return new GuardedInvocation(Lookup.emptyGetter(returnType), getMap().getProtoGetSwitchPoint(proto, name), guard); } private static GuardedInvocation findMegaMorphicGetMethod(final CallSiteDescriptor desc, final String name) { @@ -1822,27 +1824,31 @@ public abstract class ScriptObject extends PropertyListenerManager implements Pr } assert canBeFastScope || !NashornCallSiteDescriptor.isFastScope(desc); final PropertyMap myMap = getMap(); - return new GuardedInvocation(Lookup.EMPTY_SETTER, myMap.getProtoGetSwitchPoint(name), NashornGuards.getMapGuard(myMap)); + return new GuardedInvocation(Lookup.EMPTY_SETTER, myMap.getProtoGetSwitchPoint(proto, name), NashornGuards.getMapGuard(myMap)); } @SuppressWarnings("unused") - private static void setEmbed(final CallSiteDescriptor desc, final PropertyMap oldMap, final PropertyMap newMap, final MethodHandle setter, final int i, final Object self, final Object value) throws Throwable { + private static void setField(final CallSiteDescriptor desc, final PropertyMap oldMap, final PropertyMap newMap, final MethodHandle setter, final Object self, final Object value) throws Throwable { final ScriptObject obj = (ScriptObject)self; - if (obj.trySetEmbedOrSpill(desc, oldMap, newMap, value)) { - obj.useEmbed(i); + final boolean isStrict = NashornCallSiteDescriptor.isStrict(desc); + if (!obj.isExtensible()) { + throw typeError("object.non.extensible", desc.getNameToken(2), ScriptRuntime.safeToString(obj)); + } else if (obj.compareAndSetMap(oldMap, newMap)) { setter.invokeExact(self, value); + } else { + obj.set(desc.getNameToken(CallSiteDescriptor.NAME_OPERAND), value, isStrict); } } @SuppressWarnings("unused") private static void setSpill(final CallSiteDescriptor desc, final PropertyMap oldMap, final PropertyMap newMap, final int index, final Object self, final Object value) { final ScriptObject obj = (ScriptObject)self; - if (obj.trySetEmbedOrSpill(desc, oldMap, newMap, value)) { + if (obj.trySetSpill(desc, oldMap, newMap, value)) { obj.spill[index] = value; } } - private boolean trySetEmbedOrSpill(final CallSiteDescriptor desc, final PropertyMap oldMap, final PropertyMap newMap, final Object value) { + private boolean trySetSpill(final CallSiteDescriptor desc, final PropertyMap oldMap, final PropertyMap newMap, final Object value) { final boolean isStrict = NashornCallSiteDescriptor.isStrict(desc); if (!isExtensible() && isStrict) { throw typeError("object.non.extensible", desc.getNameToken(2), ScriptRuntime.safeToString(this)); @@ -1964,7 +1970,7 @@ public abstract class ScriptObject extends PropertyListenerManager implements Pr methodHandle = bindTo(methodHandle, UNDEFINED); } return new GuardedInvocation(methodHandle, - find.isInherited()? getMap().getProtoGetSwitchPoint(NO_SUCH_PROPERTY_NAME) : null, + find.isInherited()? getMap().getProtoGetSwitchPoint(proto, NO_SUCH_PROPERTY_NAME) : null, getKnownFunctionPropertyGuard(getMap(), find.getGetter(Object.class), find.getOwner(), func)); } } @@ -1995,7 +2001,7 @@ public abstract class ScriptObject extends PropertyListenerManager implements Pr } private GuardedInvocation createEmptyGetter(final CallSiteDescriptor desc, final String name) { - return new GuardedInvocation(Lookup.emptyGetter(desc.getMethodType().returnType()), getMap().getProtoGetSwitchPoint(name), NashornGuards.getMapGuard(getMap())); + return new GuardedInvocation(Lookup.emptyGetter(desc.getMethodType().returnType()), getMap().getProtoGetSwitchPoint(proto, name), NashornGuards.getMapGuard(getMap())); } private abstract static class ScriptObjectIterator implements Iterator { @@ -2070,36 +2076,39 @@ public abstract class ScriptObject extends PropertyListenerManager implements Pr * @return Added property. */ private Property addSpillProperty(final String key, final int propertyFlags) { - int i = findEmbed(); - Property spillProperty; + int fieldCount = getMap().getFieldCount(); + int fieldMaximum = getMap().getFieldMaximum(); + Property property; - if (i >= EMBED_SIZE) { - i = getMap().getSpillLength(); + if (fieldCount < fieldMaximum) { + property = new AccessorProperty(key, propertyFlags & ~Property.IS_SPILL, getClass(), fieldCount); + notifyPropertyAdded(this, property); + property = addOwnProperty(property); + } else { + int i = getMap().getSpillLength(); MethodHandle getter = MH.arrayElementGetter(Object[].class); MethodHandle setter = MH.arrayElementSetter(Object[].class); getter = MH.asType(MH.insertArguments(getter, 1, i), Lookup.GET_OBJECT_TYPE); setter = MH.asType(MH.insertArguments(setter, 1, i), Lookup.SET_OBJECT_TYPE); - spillProperty = new SpillProperty(key, propertyFlags | Property.IS_SPILL, i, getter, setter); - notifyPropertyAdded(this, spillProperty); - spillProperty = addOwnProperty(spillProperty); - i = spillProperty.getSlot(); + property = new AccessorProperty(key, propertyFlags | Property.IS_SPILL, i, getter, setter); + notifyPropertyAdded(this, property); + property = addOwnProperty(property); + i = property.getSlot(); final int newLength = (i + SPILL_RATE) / SPILL_RATE * SPILL_RATE; - final Object[] newSpill = new Object[newLength]; - if (spill != null) { - System.arraycopy(spill, 0, newSpill, 0, spill.length); + if (spill == null || newLength > spill.length) { + final Object[] newSpill = new Object[newLength]; + + if (spill != null) { + System.arraycopy(spill, 0, newSpill, 0, spill.length); + } + + spill = newSpill; } - - spill = newSpill; - } else { - useEmbed(i); - spillProperty = new SpillProperty(key, propertyFlags, i, GET_EMBED[i], SET_EMBED[i]); - notifyPropertyAdded(this, spillProperty); - spillProperty = addOwnProperty(spillProperty); } - return spillProperty; + return property; } @@ -3158,41 +3167,6 @@ public abstract class ScriptObject extends PropertyListenerManager implements Pr return true; } - /* - * Embed management - */ - - /** Number of embed slots */ - public static final int EMBED_SIZE = 4; - /** Embed offset */ - public static final int EMBED_OFFSET = 32 - EMBED_SIZE; - - static final MethodHandle[] GET_EMBED; - static final MethodHandle[] SET_EMBED; - - static { - GET_EMBED = new MethodHandle[EMBED_SIZE]; - SET_EMBED = new MethodHandle[EMBED_SIZE]; - - for (int i = 0; i < EMBED_SIZE; i++) { - final String name = "embed" + i; - GET_EMBED[i] = MH.asType(MH.getter(MethodHandles.lookup(), ScriptObject.class, name, Object.class), Lookup.GET_OBJECT_TYPE); - SET_EMBED[i] = MH.asType(MH.setter(MethodHandles.lookup(), ScriptObject.class, name, Object.class), Lookup.SET_OBJECT_TYPE); - } - } - - void useEmbed(final int i) { - flags |= 1 << (EMBED_OFFSET + i); - } - - int findEmbed() { - final int bits = ~(flags >>> EMBED_OFFSET); - final int least = bits ^ -bits; - final int index = Integer.numberOfTrailingZeros(least) - 1; - - return index; - } - /* * Make a new UserAccessorProperty property. getter and setter functions are stored in * this ScriptObject and slot values are used in property object. @@ -3200,26 +3174,16 @@ public abstract class ScriptObject extends PropertyListenerManager implements Pr private UserAccessorProperty newUserAccessors(final String key, final int propertyFlags, final ScriptFunction getter, final ScriptFunction setter) { int oldSpillLength = getMap().getSpillLength(); - int getterSlot = findEmbed(); - if (getterSlot >= EMBED_SIZE) { - getterSlot = oldSpillLength + EMBED_SIZE; - ++oldSpillLength; - } else { - useEmbed(getterSlot); - } - setEmbedOrSpill(getterSlot, getter); + int getterSlot = oldSpillLength++; + setSpill(getterSlot, getter); // if getter function is null, flag the slot to be negative (less by 1) if (getter == null) { getterSlot = -getterSlot - 1; } - int setterSlot = findEmbed(); - if (setterSlot >= EMBED_SIZE) { - setterSlot = oldSpillLength + EMBED_SIZE; - } else { - useEmbed(setterSlot); - } - setEmbedOrSpill(setterSlot, setter); + int setterSlot = oldSpillLength++; + + setSpill(setterSlot, setter); // if setter function is null, flag the slot to be negative (less by 1) if (setter == null) { setterSlot = -setterSlot - 1; @@ -3228,56 +3192,28 @@ public abstract class ScriptObject extends PropertyListenerManager implements Pr return new UserAccessorProperty(key, propertyFlags, getterSlot, setterSlot); } - private void setEmbedOrSpill(final int slot, final Object value) { - switch (slot) { - case 0: - embed0 = value; - break; - case 1: - embed1 = value; - break; - case 2: - embed2 = value; - break; - case 3: - embed3 = value; - break; - default: - if (slot >= 0) { - final int index = (slot - EMBED_SIZE); - if (spill == null) { - // create new spill. - spill = new Object[Math.max(index + 1, SPILL_RATE)]; - } else if (index >= spill.length) { - // grow spill as needed - final Object[] newSpill = new Object[index + 1]; - System.arraycopy(spill, 0, newSpill, 0, spill.length); - spill = newSpill; - } - - spill[index] = value; + private void setSpill(final int slot, final Object value) { + if (slot >= 0) { + final int index = slot; + if (spill == null) { + // create new spill. + spill = new Object[Math.max(index + 1, SPILL_RATE)]; + } else if (index >= spill.length) { + // grow spill as needed + final Object[] newSpill = new Object[index + 1]; + System.arraycopy(spill, 0, newSpill, 0, spill.length); + spill = newSpill; } - break; + + spill[index] = value; } } - // user accessors are either stored in embed fields or spill array slots - // get the accessor value using slot number. Note that slot is either embed - // field number or (spill array index + embedSize). - Object getEmbedOrSpill(final int slot) { - switch (slot) { - case 0: - return embed0; - case 1: - return embed1; - case 2: - return embed2; - case 3: - return embed3; - default: - final int index = (slot - EMBED_SIZE); - return (index < 0 || (index >= spill.length)) ? null : spill[index]; - } + // user accessors are either stored in spill array slots + // get the accessor value using slot number. Note that slot is spill array index. + Object getSpill(final int slot) { + final int index = slot; + return (index < 0 || (index >= spill.length)) ? null : spill[index]; } // User defined getter and setter are always called by "dyn:call". Note that the user @@ -3287,7 +3223,7 @@ public abstract class ScriptObject extends PropertyListenerManager implements Pr @SuppressWarnings("unused") private static Object userAccessorGetter(final ScriptObject proto, final int slot, final Object self) { final ScriptObject container = (proto != null) ? proto : (ScriptObject)self; - final Object func = container.getEmbedOrSpill(slot); + final Object func = container.getSpill(slot); if (func instanceof ScriptFunction) { try { @@ -3305,7 +3241,7 @@ public abstract class ScriptObject extends PropertyListenerManager implements Pr @SuppressWarnings("unused") private static void userAccessorSetter(final ScriptObject proto, final int slot, final String name, final Object self, final Object value) { final ScriptObject container = (proto != null) ? proto : (ScriptObject)self; - final Object func = container.getEmbedOrSpill(slot); + final Object func = container.getSpill(slot); if (func instanceof ScriptFunction) { try { diff --git a/nashorn/src/jdk/nashorn/internal/runtime/SetMethodCreator.java b/nashorn/src/jdk/nashorn/internal/runtime/SetMethodCreator.java index 8012ce02ec4..5ff321fc9a0 100644 --- a/nashorn/src/jdk/nashorn/internal/runtime/SetMethodCreator.java +++ b/nashorn/src/jdk/nashorn/internal/runtime/SetMethodCreator.java @@ -166,18 +166,20 @@ final class SetMethodCreator { } private SetMethod createNewPropertySetter() { - final int nextEmbed = sobj.findEmbed(); - final SetMethod sm; - if (nextEmbed >= ScriptObject.EMBED_SIZE) { - sm = createNewSpillPropertySetter(); - } else { - sm = createNewEmbedPropertySetter(nextEmbed); - } - + final SetMethod sm = map.getFieldCount() < map.getFieldMaximum() ? createNewFieldSetter() : createNewSpillPropertySetter(); sobj.notifyPropertyAdded(sobj, sm.property); return sm; } + private SetMethod createNewFieldSetter() { + final PropertyMap oldMap = getMap(); + final Property property = new AccessorProperty(getName(), 0, sobj.getClass(), oldMap.getFieldCount()); + final PropertyMap newMap = oldMap.addProperty(property); + MethodHandle setter = MH.insertArguments(ScriptObject.SETFIELD, 0, desc, oldMap, newMap, property.getSetter(Object.class, newMap)); + + return new SetMethod(MH.asType(setter, Lookup.SET_OBJECT_TYPE), property); + } + private SetMethod createNewSpillPropertySetter() { final int nextSpill = getMap().getSpillLength(); @@ -189,7 +191,7 @@ final class SetMethodCreator { final MethodHandle getter = MH.asType(MH.insertArguments(MH.arrayElementGetter(Object[].class), 1, nextSpill), Lookup.GET_OBJECT_TYPE); final MethodHandle setter = MH.asType(MH.insertArguments(MH.arrayElementSetter(Object[].class), 1, nextSpill), Lookup.SET_OBJECT_TYPE); - return new SpillProperty(getName(), Property.IS_SPILL, nextSpill, getter, setter); + return new AccessorProperty(getName(), Property.IS_SPILL, nextSpill, getter, setter); } private MethodHandle createSpillMethodHandle(final int nextSpill, Property property) { @@ -207,14 +209,6 @@ final class SetMethodCreator { } } - private SetMethod createNewEmbedPropertySetter(final int nextEmbed) { - sobj.useEmbed(nextEmbed); - final Property property = new SpillProperty(getName(), 0, nextEmbed, ScriptObject.GET_EMBED[nextEmbed], ScriptObject.SET_EMBED[nextEmbed]); - //TODO specfields - final MethodHandle methodHandle = MH.insertArguments(ScriptObject.SETEMBED, 0, desc, getMap(), getNewMap(property), property.getSetter(Object.class, getMap()), nextEmbed); - return new SetMethod(methodHandle, property); - } - private PropertyMap getNewMap(Property property) { return getMap().addProperty(property); } diff --git a/nashorn/src/jdk/nashorn/internal/runtime/SpillProperty.java b/nashorn/src/jdk/nashorn/internal/runtime/SpillProperty.java deleted file mode 100644 index 716097d00d5..00000000000 --- a/nashorn/src/jdk/nashorn/internal/runtime/SpillProperty.java +++ /dev/null @@ -1,85 +0,0 @@ -/* - * Copyright (c) 2010, 2013, Oracle and/or its affiliates. All rights reserved. - * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. - * - * This code is free software; you can redistribute it and/or modify it - * under the terms of the GNU General Public License version 2 only, as - * published by the Free Software Foundation. Oracle designates this - * particular file as subject to the "Classpath" exception as provided - * by Oracle in the LICENSE file that accompanied this code. - * - * This code is distributed in the hope that it will be useful, but WITHOUT - * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or - * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License - * version 2 for more details (a copy is included in the LICENSE file that - * accompanied this code). - * - * You should have received a copy of the GNU General Public License version - * 2 along with this work; if not, write to the Free Software Foundation, - * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. - * - * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA - * or visit www.oracle.com if you need additional information or have any - * questions. - */ - -package jdk.nashorn.internal.runtime; - -import static jdk.nashorn.internal.lookup.Lookup.MH; - -import java.lang.invoke.MethodHandle; -import java.lang.invoke.MethodHandles; -import jdk.nashorn.internal.lookup.Lookup; - -/** - * The SpillProperty is a subclass of AccessorProperties. Anything not in the initial property map - * will end up in the embed fields of the ScriptObject or in the Spill, which currently is a growing - * Object only array in ScriptObject - * - * @see AccessorProperty - * @see ScriptObject - */ -public final class SpillProperty extends AccessorProperty { - private static final MethodHandle SPILLGETTER = MH.asType(MH.getter(MethodHandles.lookup(), ScriptObject.class, "spill", Object[].class), Lookup.GET_OBJECT_TYPE); - - /** - * Constructor - * - * @param key property key - * @param flags property flags - * @param slot property slot/index - * @param getter getter for property - * @param setter setter for property, or null if not configurable and writable - */ - public SpillProperty(final String key, final int flags, final int slot, final MethodHandle getter, final MethodHandle setter) { - super(key, flags, slot, getter, setter); - } - - private SpillProperty(final SpillProperty property) { - super(property); - } - - @Override - protected Property copy() { - return new SpillProperty(this); - } - - @Override - public MethodHandle getGetter(final Class type) { - if (isSpill()) { - return MH.filterArguments(super.getGetter(type), 0, SPILLGETTER); - } - - return super.getGetter(type); - } - - @Override - public MethodHandle getSetter(final Class type, final PropertyMap currentMap) { - if (isSpill()) { - return MH.filterArguments(super.getSetter(type, currentMap), 0, SPILLGETTER); - } - - return super.getSetter(type, currentMap); - } - -} diff --git a/nashorn/src/jdk/nashorn/internal/runtime/StructureLoader.java b/nashorn/src/jdk/nashorn/internal/runtime/StructureLoader.java index db55fff963a..ff6973a9fc1 100644 --- a/nashorn/src/jdk/nashorn/internal/runtime/StructureLoader.java +++ b/nashorn/src/jdk/nashorn/internal/runtime/StructureLoader.java @@ -110,8 +110,7 @@ final class StructureLoader extends NashornLoader { @Override protected Class findClass(final String name) throws ClassNotFoundException { if (name.startsWith(JS_OBJECT_PREFIX_EXTERNAL)) { - final int start = name.indexOf(JS_OBJECT_PREFIX.symbolName()) + JS_OBJECT_PREFIX.symbolName().length(); - return generateClass(name, name.substring(start, name.length())); + return generateClass(name, name.substring(JS_OBJECT_PREFIX_EXTERNAL.length())); } return super.findClass(name); } diff --git a/nashorn/src/jdk/nashorn/internal/runtime/UserAccessorProperty.java b/nashorn/src/jdk/nashorn/internal/runtime/UserAccessorProperty.java index 83d3c8d7152..75c285a6fc8 100644 --- a/nashorn/src/jdk/nashorn/internal/runtime/UserAccessorProperty.java +++ b/nashorn/src/jdk/nashorn/internal/runtime/UserAccessorProperty.java @@ -67,7 +67,6 @@ public final class UserAccessorProperty extends Property { private UserAccessorProperty(final UserAccessorProperty property) { super(property); - this.getterSlot = property.getterSlot; this.setterSlot = property.setterSlot; } @@ -115,10 +114,10 @@ public final class UserAccessorProperty extends Property { public int getSpillCount() { // calculate how many spill array slots used by this propery. int count = 0; - if (getGetterSlot() >= ScriptObject.EMBED_SIZE) { + if (getGetterSlot() >= 0) { count++; } - if (getSetterSlot() >= ScriptObject.EMBED_SIZE) { + if (getSetterSlot() >= 0) { count++; } return count; @@ -141,7 +140,7 @@ public final class UserAccessorProperty extends Property { @Override public ScriptFunction getGetterFunction(final ScriptObject obj) { - final Object value = obj.getEmbedOrSpill(getterSlot); + final Object value = obj.getSpill(getterSlot); return (value instanceof ScriptFunction) ? (ScriptFunction) value : null; } @@ -152,7 +151,7 @@ public final class UserAccessorProperty extends Property { @Override public ScriptFunction getSetterFunction(final ScriptObject obj) { - final Object value = obj.getEmbedOrSpill(setterSlot); + final Object value = obj.getSpill(setterSlot); return (value instanceof ScriptFunction) ? (ScriptFunction) value : null; } diff --git a/nashorn/src/jdk/nashorn/internal/scripts/JO.java b/nashorn/src/jdk/nashorn/internal/scripts/JO.java index 44412364113..d698a2637ce 100644 --- a/nashorn/src/jdk/nashorn/internal/scripts/JO.java +++ b/nashorn/src/jdk/nashorn/internal/scripts/JO.java @@ -32,12 +32,11 @@ import jdk.nashorn.internal.runtime.ScriptObject; * Empty object class. */ public class JO extends ScriptObject { - /** * Constructor */ public JO() { - super(); + super(PropertyMap.newMap(JO.class)); } /** diff --git a/nashorn/src/jdk/nashorn/tools/Shell.java b/nashorn/src/jdk/nashorn/tools/Shell.java index 22d879f5f3b..20725c0d3f2 100644 --- a/nashorn/src/jdk/nashorn/tools/Shell.java +++ b/nashorn/src/jdk/nashorn/tools/Shell.java @@ -343,7 +343,7 @@ public class Shell { * @return error code * @throws IOException when any script file read results in I/O error */ - private int runFXScripts(final Context context, final ScriptObject global, final List files) throws IOException { + private static int runFXScripts(final Context context, final ScriptObject global, final List files) throws IOException { final ScriptObject oldGlobal = Context.getGlobal(); final boolean globalChanged = (oldGlobal != global); try { From 323a923236da7ebfb3d158d5e45438b95b3d35e2 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Hannes=20Walln=C3=B6fer?= Date: Thu, 2 May 2013 09:19:44 +0200 Subject: [PATCH 02/15] 8013729: SwitchPoint invalidation not working over prototype chain Reviewed-by: lagergren, sundar --- .../internal/runtime/ScriptObject.java | 3 ++ nashorn/test/script/basic/JDK-8013729.js | 49 +++++++++++++++++++ .../test/script/basic/JDK-8013729.js.EXPECTED | 4 ++ 3 files changed, 56 insertions(+) create mode 100644 nashorn/test/script/basic/JDK-8013729.js create mode 100644 nashorn/test/script/basic/JDK-8013729.js.EXPECTED diff --git a/nashorn/src/jdk/nashorn/internal/runtime/ScriptObject.java b/nashorn/src/jdk/nashorn/internal/runtime/ScriptObject.java index 18724fb1cad..630bbbd0ca4 100644 --- a/nashorn/src/jdk/nashorn/internal/runtime/ScriptObject.java +++ b/nashorn/src/jdk/nashorn/internal/runtime/ScriptObject.java @@ -1325,6 +1325,9 @@ public abstract class ScriptObject extends PropertyListenerManager implements Pr * Flag this object as having a prototype. */ public void setIsPrototype() { + if (proto != null && !isPrototype()) { + proto.addPropertyListener(this); + } flags |= IS_PROTOTYPE; } diff --git a/nashorn/test/script/basic/JDK-8013729.js b/nashorn/test/script/basic/JDK-8013729.js new file mode 100644 index 00000000000..a500602a953 --- /dev/null +++ b/nashorn/test/script/basic/JDK-8013729.js @@ -0,0 +1,49 @@ +/* + * Copyright (c) 2010, 2013, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/** + * JDK-8013729: SwitchPoint invalidation not working over prototype chain + * + * @test + * @run + */ + +var a = {x: 1}; +var b = Object.create(a, {y: {value: 2, configurable: true, writable: true}}); +var c = Object.create(b); +var d = Object.create(c); + +function p() { + print(d.x); + print(d.y); +} + +p(); + +// Delete, then redefine properties +delete a.x; +delete b.y; +a.x = 10; +b.y = 20; + +p(); diff --git a/nashorn/test/script/basic/JDK-8013729.js.EXPECTED b/nashorn/test/script/basic/JDK-8013729.js.EXPECTED new file mode 100644 index 00000000000..c9fc4c6113a --- /dev/null +++ b/nashorn/test/script/basic/JDK-8013729.js.EXPECTED @@ -0,0 +1,4 @@ +1 +2 +10 +20 From 6ee68537b6080945cf79c7b793ba9c3982ced058 Mon Sep 17 00:00:00 2001 From: James Laskey Date: Thu, 2 May 2013 13:22:46 -0300 Subject: [PATCH 03/15] 8013794: JDK-8006220 caused an octane performance regression Reviewed-by: lagergren, sundar --- nashorn/src/jdk/nashorn/internal/codegen/ObjectCreator.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/nashorn/src/jdk/nashorn/internal/codegen/ObjectCreator.java b/nashorn/src/jdk/nashorn/internal/codegen/ObjectCreator.java index 82efd14c4fa..355dc6fdafb 100644 --- a/nashorn/src/jdk/nashorn/internal/codegen/ObjectCreator.java +++ b/nashorn/src/jdk/nashorn/internal/codegen/ObjectCreator.java @@ -92,7 +92,7 @@ public abstract class ObjectCreator { } } - paddedFieldCount = (fieldCount + FIELD_PADDING + FIELD_ROUNDING - 1) / FIELD_ROUNDING * FIELD_ROUNDING; + paddedFieldCount = fieldCount + FIELD_PADDING; } /** From cd9c2c1bb247dfb4c5e03837466c6a181c8192a7 Mon Sep 17 00:00:00 2001 From: James Laskey Date: Thu, 2 May 2013 15:01:16 -0300 Subject: [PATCH 04/15] 8013796: load("fx:base.js") should not be in fx:bootstrap.js Reviewed-by: sundar, lagergren --- .../src/jdk/nashorn/internal/runtime/resources/fx/bootstrap.js | 2 -- 1 file changed, 2 deletions(-) diff --git a/nashorn/src/jdk/nashorn/internal/runtime/resources/fx/bootstrap.js b/nashorn/src/jdk/nashorn/internal/runtime/resources/fx/bootstrap.js index 97b01a0871f..b78ebbc8e9d 100644 --- a/nashorn/src/jdk/nashorn/internal/runtime/resources/fx/bootstrap.js +++ b/nashorn/src/jdk/nashorn/internal/runtime/resources/fx/bootstrap.js @@ -35,8 +35,6 @@ com.sun.javafx.application.LauncherImpl.launchApplication((Java.extend(javafx.ap init: function() { // Java FX packages and classes must be defined here because // they may not be viable until launch time due to clinit ordering. - - load("fx:base.js"); }, // Overridden javafx.application.Application.start(Stage stage); From 6f6ec2d9d10e78e9dbf604acfaf075187c95bc97 Mon Sep 17 00:00:00 2001 From: Marcus Lagergren Date: Fri, 3 May 2013 15:33:54 +0200 Subject: [PATCH 05/15] 8013477: Node.setSymbol needs to be copy on write - enable IR snapshots for recompilation based on callsite type specialization. [not enabled by default, hidden by a flag for now] Reviewed-by: jlaskey, hannesw --- nashorn/bin/jjs | 2 +- .../scripting/NashornScriptEngineFactory.java | 1 - .../jdk/nashorn/internal/codegen/Attr.java | 583 +++++++++--------- .../internal/codegen/CodeGenerator.java | 27 +- .../internal/codegen/CompilationPhase.java | 31 +- .../nashorn/internal/codegen/Compiler.java | 111 ++-- .../internal/codegen/FinalizeTypes.java | 21 +- .../internal/codegen/ObjectCreator.java | 1 - .../nashorn/internal/codegen/Splitter.java | 8 +- .../src/jdk/nashorn/internal/ir/Block.java | 45 +- .../jdk/nashorn/internal/ir/CatchNode.java | 7 +- .../jdk/nashorn/internal/ir/FunctionNode.java | 188 ++++-- .../nashorn/internal/ir/LexicalContext.java | 3 - .../internal/ir/LexicalContextNode.java | 12 + .../jdk/nashorn/internal/ir/LiteralNode.java | 29 +- nashorn/src/jdk/nashorn/internal/ir/Node.java | 13 +- .../src/jdk/nashorn/internal/ir/Symbol.java | 11 + .../internal/objects/NativeRegExp.java | 32 +- .../internal/parser/AbstractParser.java | 1 + .../jdk/nashorn/internal/parser/Parser.java | 20 +- .../internal/runtime/CompiledFunction.java | 2 +- .../internal/runtime/CompiledFunctions.java | 1 - .../jdk/nashorn/internal/runtime/Context.java | 8 +- .../internal/runtime/JSONFunctions.java | 2 - .../RecompilableScriptFunctionData.java | 122 +++- .../internal/runtime/ScriptEnvironment.java | 30 + .../internal/runtime/ScriptObject.java | 6 +- .../runtime/regexp/DefaultRegExp.java | 20 +- .../internal/runtime/regexp/JoniRegExp.java | 20 +- .../runtime/regexp/RegExpScanner.java | 4 +- .../runtime/resources/Options.properties | 20 +- nashorn/src/jdk/nashorn/tools/Shell.java | 2 +- nashorn/test/script/basic/paramspec.js | 38 ++ .../test/script/basic/paramspec.js.EXPECTED | 2 + nashorn/test/script/basic/runsunspider.js | 2 +- .../logcoverage.js | 5 +- 36 files changed, 830 insertions(+), 600 deletions(-) create mode 100644 nashorn/test/script/basic/paramspec.js create mode 100644 nashorn/test/script/basic/paramspec.js.EXPECTED rename nashorn/test/script/{trusted => currently-failing}/logcoverage.js (94%) diff --git a/nashorn/bin/jjs b/nashorn/bin/jjs index fe6665c3c3d..f89a07c2c9a 100644 --- a/nashorn/bin/jjs +++ b/nashorn/bin/jjs @@ -26,4 +26,4 @@ [ -z "$JAVA_HOME" ] && echo "Please set JAVA_HOME" && exit 1; -$JAVA_HOME/bin/java -server -XX:-TieredCompilation -Xms2G -Xmx2G -esa -ea -Djava.ext.dirs=`dirname $0`/../dist:$JAVA_HOME/jre/lib/ext -XX:+HeapDumpOnOutOfMemoryError -Djava.lang.invoke.MethodHandle.DEBUG_NAMES=false -Dnashorn.debug=true jdk.nashorn.tools.Shell $* +$JAVA_HOME/bin/java -server -XX:+TieredCompilation -Xms2G -Xmx2G -esa -ea -Djava.ext.dirs=`dirname $0`/../dist:$JAVA_HOME/jre/lib/ext -XX:+HeapDumpOnOutOfMemoryError -Djava.lang.invoke.MethodHandle.DEBUG_NAMES=false -Dnashorn.debug=true jdk.nashorn.tools.Shell $* diff --git a/nashorn/src/jdk/nashorn/api/scripting/NashornScriptEngineFactory.java b/nashorn/src/jdk/nashorn/api/scripting/NashornScriptEngineFactory.java index 1ca6dcdd4b3..c10054158fe 100644 --- a/nashorn/src/jdk/nashorn/api/scripting/NashornScriptEngineFactory.java +++ b/nashorn/src/jdk/nashorn/api/scripting/NashornScriptEngineFactory.java @@ -31,7 +31,6 @@ import java.util.List; import javax.script.ScriptEngine; import javax.script.ScriptEngineFactory; import jdk.nashorn.internal.runtime.Version; -import sun.reflect.Reflection; /** * JSR-223 compliant script engine factory for Nashorn. The engine answers for: diff --git a/nashorn/src/jdk/nashorn/internal/codegen/Attr.java b/nashorn/src/jdk/nashorn/internal/codegen/Attr.java index 9b17c47f1a5..d5c9ef07a60 100644 --- a/nashorn/src/jdk/nashorn/internal/codegen/Attr.java +++ b/nashorn/src/jdk/nashorn/internal/codegen/Attr.java @@ -29,17 +29,21 @@ import static jdk.nashorn.internal.codegen.CompilerConstants.ARGUMENTS; import static jdk.nashorn.internal.codegen.CompilerConstants.CALLEE; import static jdk.nashorn.internal.codegen.CompilerConstants.EXCEPTION_PREFIX; import static jdk.nashorn.internal.codegen.CompilerConstants.ITERATOR_PREFIX; +import static jdk.nashorn.internal.codegen.CompilerConstants.LITERAL_PREFIX; import static jdk.nashorn.internal.codegen.CompilerConstants.RETURN; import static jdk.nashorn.internal.codegen.CompilerConstants.SCOPE; import static jdk.nashorn.internal.codegen.CompilerConstants.SWITCH_TAG_PREFIX; +import static jdk.nashorn.internal.codegen.CompilerConstants.TEMP_PREFIX; import static jdk.nashorn.internal.codegen.CompilerConstants.THIS; import static jdk.nashorn.internal.codegen.CompilerConstants.VARARGS; +import static jdk.nashorn.internal.ir.Symbol.IS_CONSTANT; import static jdk.nashorn.internal.ir.Symbol.IS_FUNCTION_SELF; import static jdk.nashorn.internal.ir.Symbol.IS_GLOBAL; import static jdk.nashorn.internal.ir.Symbol.IS_INTERNAL; import static jdk.nashorn.internal.ir.Symbol.IS_LET; import static jdk.nashorn.internal.ir.Symbol.IS_PARAM; import static jdk.nashorn.internal.ir.Symbol.IS_SCOPE; +import static jdk.nashorn.internal.ir.Symbol.IS_TEMP; import static jdk.nashorn.internal.ir.Symbol.IS_THIS; import static jdk.nashorn.internal.ir.Symbol.IS_VAR; import static jdk.nashorn.internal.ir.Symbol.KINDMASK; @@ -51,6 +55,7 @@ import java.util.HashSet; import java.util.Iterator; import java.util.List; import java.util.Set; + import jdk.nashorn.internal.codegen.types.Type; import jdk.nashorn.internal.ir.AccessNode; import jdk.nashorn.internal.ir.BinaryNode; @@ -90,7 +95,6 @@ import jdk.nashorn.internal.runtime.ECMAException; import jdk.nashorn.internal.runtime.JSType; import jdk.nashorn.internal.runtime.Property; import jdk.nashorn.internal.runtime.PropertyMap; -import jdk.nashorn.internal.runtime.ScriptFunction; import jdk.nashorn.internal.runtime.ScriptObject; /** @@ -133,9 +137,9 @@ final class Attr extends NodeOperatorVisitor { * Constructor. */ Attr() { - localDefs = new ArrayDeque<>(); - localUses = new ArrayDeque<>(); - returnTypes = new ArrayDeque<>(); + this.localDefs = new ArrayDeque<>(); + this.localUses = new ArrayDeque<>(); + this.returnTypes = new ArrayDeque<>(); } @Override @@ -150,67 +154,48 @@ final class Attr extends NodeOperatorVisitor { @Override public Node leaveAccessNode(final AccessNode accessNode) { - ensureSymbol(Type.OBJECT, accessNode); //While Object type is assigned here, Access Specialization in FinalizeTypes may narrow this - end(accessNode); - return accessNode; + //While Object type is assigned here, Access Specialization in FinalizeTypes may narrow this + return end(ensureSymbol(Type.OBJECT, accessNode)); } - private void enterFunctionBody() { + private void initFunctionWideVariables(final FunctionNode functionNode, final Block body) { + initCompileConstant(CALLEE, body, IS_PARAM | IS_INTERNAL, FunctionNode.FUNCTION_TYPE); + initCompileConstant(THIS, body, IS_PARAM | IS_THIS, Type.OBJECT); - final FunctionNode functionNode = getLexicalContext().getCurrentFunction(); - final Block body = getLexicalContext().getCurrentBlock(); - initCallee(body); - initThis(body); if (functionNode.isVarArg()) { - initVarArg(body, functionNode.needsArguments()); + initCompileConstant(VARARGS, body, IS_PARAM | IS_INTERNAL, Type.OBJECT_ARRAY); + if (functionNode.needsArguments()) { + initCompileConstant(ARGUMENTS, body, IS_VAR | IS_INTERNAL, Type.typeFor(ScriptObject.class)); + addLocalDef(ARGUMENTS.symbolName()); + } } initParameters(functionNode, body); - initScope(body); - initReturn(body); + initCompileConstant(SCOPE, body, IS_VAR | IS_INTERNAL, Type.typeFor(ScriptObject.class)); + initCompileConstant(RETURN, body, IS_VAR | IS_INTERNAL, Type.OBJECT); + } - if (functionNode.isProgram()) { - initFromPropertyMap(body); - } else if(!functionNode.isDeclared()) { - // It's neither declared nor program - it's a function expression then; assign it a self-symbol. - - if (functionNode.getSymbol() != null) { - // a temporary left over from an earlier pass when the function was lazy - assert functionNode.getSymbol().isTemp(); - // remove it - functionNode.setSymbol(null); - } - final boolean anonymous = functionNode.isAnonymous(); - final String name = anonymous ? null : functionNode.getIdent().getName(); - if (anonymous || body.getExistingSymbol(name) != null) { - // The function is either anonymous, or another local identifier already trumps its name on entry: - // either it has the same name as one of its parameters, or is named "arguments" and also references the - // "arguments" identifier in its body. - ensureSymbol(functionNode, Type.typeFor(ScriptFunction.class), functionNode); - } else { - final Symbol selfSymbol = defineSymbol(body, name, IS_VAR | IS_FUNCTION_SELF, functionNode); - assert selfSymbol.isFunctionSelf(); - newType(selfSymbol, Type.OBJECT); - } - } - - /* - * This pushes all declarations (except for non-statements, i.e. for - * node temporaries) to the top of the function scope. This way we can - * get around problems like - * - * while (true) { - * break; - * if (true) { - * var s; - * } - * } - * - * to an arbitrary nesting depth. - * - * @see NASHORN-73 - */ + /** + * This pushes all declarations (except for non-statements, i.e. for + * node temporaries) to the top of the function scope. This way we can + * get around problems like + * + * while (true) { + * break; + * if (true) { + * var s; + * } + * } + * + * to an arbitrary nesting depth. + * + * see NASHORN-73 + * + * @param functionNode the FunctionNode we are entering + * @param body the body of the FunctionNode we are entering + */ + private void acceptDeclarations(final FunctionNode functionNode, final Block body) { // This visitor will assign symbol to all declared variables, except function declarations (which are taken care // in a separate step above) and "var" declarations in for loop initializers. body.accept(new NodeOperatorVisitor() { @@ -220,27 +205,52 @@ final class Attr extends NodeOperatorVisitor { } @Override - public boolean enterVarNode(final VarNode varNode) { - + public Node leaveVarNode(final VarNode varNode) { // any declared symbols that aren't visited need to be typed as well, hence the list - if (varNode.isStatement()) { - - final IdentNode ident = varNode.getName(); - final Symbol symbol = defineSymbol(body, ident.getName(), IS_VAR, new IdentNode(ident)); + final IdentNode ident = varNode.getName(); + final Symbol symbol = defineSymbol(body, ident.getName(), IS_VAR); functionNode.addDeclaredSymbol(symbol); if (varNode.isFunctionDeclaration()) { newType(symbol, FunctionNode.FUNCTION_TYPE); } + return varNode.setName((IdentNode)ident.setSymbol(getLexicalContext(), symbol)); } - return false; + return varNode; } }); } + private void enterFunctionBody() { + + final FunctionNode functionNode = getLexicalContext().getCurrentFunction(); + final Block body = getLexicalContext().getCurrentBlock(); + + initFunctionWideVariables(functionNode, body); + + if (functionNode.isProgram()) { + initFromPropertyMap(body); + } else if (!functionNode.isDeclared()) { + // It's neither declared nor program - it's a function expression then; assign it a self-symbol. + assert functionNode.getSymbol() == null; + + final boolean anonymous = functionNode.isAnonymous(); + final String name = anonymous ? null : functionNode.getIdent().getName(); + if (!(anonymous || body.getExistingSymbol(name) != null)) { + assert !anonymous && name != null; + newType(defineSymbol(body, name, IS_VAR | IS_FUNCTION_SELF), Type.OBJECT); + } + } + + acceptDeclarations(functionNode, body); + } + @Override public boolean enterBlock(final Block block) { start(block); + //ensure that we don't use information from a previous compile. This is very ugly TODO + //the symbols in the block should really be stateless + block.clearSymbols(); if (getLexicalContext().isFunctionBody()) { enterFunctionBody(); @@ -257,14 +267,13 @@ final class Attr extends NodeOperatorVisitor { } @Override - public Node leaveCallNode(final CallNode callNode) { - ensureSymbol(callNode.getType(), callNode); - return end(callNode); + public boolean enterCallNode(final CallNode callNode) { + return start(callNode); } @Override - public boolean enterCallNode(final CallNode callNode) { - return start(callNode); + public Node leaveCallNode(final CallNode callNode) { + return end(ensureSymbol(callNode.getType(), callNode)); } @Override @@ -275,23 +284,31 @@ final class Attr extends NodeOperatorVisitor { start(catchNode); // define block-local exception variable - final Symbol def = defineSymbol(block, exception.getName(), IS_VAR | IS_LET, exception); + final Symbol def = defineSymbol(block, exception.getName(), IS_VAR | IS_LET); newType(def, Type.OBJECT); addLocalDef(exception.getName()); return true; } + @Override + public Node leaveCatchNode(final CatchNode catchNode) { + final IdentNode exception = catchNode.getException(); + final Block block = getLexicalContext().getCurrentBlock(); + final Symbol symbol = findSymbol(block, exception.getName()); + assert symbol != null; + return end(catchNode.setException((IdentNode)exception.setSymbol(getLexicalContext(), symbol))); + } + /** * Declare the definition of a new symbol. * * @param name Name of symbol. * @param symbolFlags Symbol flags. - * @param node Defining Node. * * @return Symbol for given name or null for redefinition. */ - private Symbol defineSymbol(final Block block, final String name, final int symbolFlags, final Node node) { + private Symbol defineSymbol(final Block block, final String name, final int symbolFlags) { int flags = symbolFlags; Symbol symbol = findSymbol(block, name); // Locate symbol. @@ -337,7 +354,7 @@ final class Attr extends NodeOperatorVisitor { // Create and add to appropriate block. symbol = new Symbol(name, flags); - symbolBlock.putSymbol(name, symbol); + symbolBlock.putSymbol(getLexicalContext(), symbol); if ((flags & Symbol.KINDMASK) != IS_GLOBAL) { symbol.setNeedsSlot(true); @@ -346,10 +363,6 @@ final class Attr extends NodeOperatorVisitor { symbol.setFlags(flags); } - if (node != null) { - node.setSymbol(symbol); - } - return symbol; } @@ -357,30 +370,22 @@ final class Attr extends NodeOperatorVisitor { public boolean enterFunctionNode(final FunctionNode functionNode) { start(functionNode, false); + if (functionNode.isLazy()) { + return false; + } + + //an outermost function in our lexical context that is not a program (runScript) + //is possible - it is a function being compiled lazily if (functionNode.isDeclared()) { final Iterator blocks = getLexicalContext().getBlocks(); if (blocks.hasNext()) { - defineSymbol( - blocks.next(), - functionNode.getIdent().getName(), - IS_VAR, - functionNode); - } else { - // Q: What's an outermost function in a lexical context that is not a program? - // A: It's a function being compiled lazily! - assert getLexicalContext().getOutermostFunction() == functionNode && !functionNode.isProgram(); + defineSymbol(blocks.next(), functionNode.getIdent().getName(), IS_VAR); } } - if (functionNode.isLazy()) { - LOG.info("LAZY: ", functionNode.getName(), " => Promoting to OBJECT"); - ensureSymbol(getLexicalContext().getCurrentFunction(), Type.OBJECT, functionNode); - end(functionNode); - return false; - } - returnTypes.push(functionNode.getReturnType()); pushLocalsFunction(); + return true; } @@ -390,8 +395,29 @@ final class Attr extends NodeOperatorVisitor { final LexicalContext lc = getLexicalContext(); + final Block body = newFunctionNode.getBody(); + + //look for this function in the parent block + if (functionNode.isDeclared()) { + final Iterator blocks = getLexicalContext().getBlocks(); + if (blocks.hasNext()) { + newFunctionNode = (FunctionNode)newFunctionNode.setSymbol(lc, findSymbol(blocks.next(), functionNode.getIdent().getName())); + } + } else if (!functionNode.isProgram()) { + final boolean anonymous = functionNode.isAnonymous(); + final String name = anonymous ? null : functionNode.getIdent().getName(); + if (anonymous || body.getExistingSymbol(name) != null) { + newFunctionNode = (FunctionNode)Attr.ensureSymbol(lc, body, FunctionNode.FUNCTION_TYPE, newFunctionNode); + } else { + assert name != null; + final Symbol self = body.getExistingSymbol(name); + assert self != null && self.isFunctionSelf(); + newFunctionNode = (FunctionNode)newFunctionNode.setSymbol(lc, body.getExistingSymbol(name)); + } + } + //unknown parameters are promoted to object type. - finalizeParameters(newFunctionNode); + newFunctionNode = finalizeParameters(newFunctionNode); finalizeTypes(newFunctionNode); for (final Symbol symbol : newFunctionNode.getDeclaredSymbols()) { if (symbol.getSymbolType().isUnknown()) { @@ -400,8 +426,6 @@ final class Attr extends NodeOperatorVisitor { } } - final Block body = newFunctionNode.getBody(); - if (newFunctionNode.hasLazyChildren()) { //the final body has already been assigned as we have left the function node block body by now objectifySymbols(body); @@ -409,7 +433,7 @@ final class Attr extends NodeOperatorVisitor { if (body.getFlag(Block.NEEDS_SELF_SYMBOL)) { final IdentNode callee = compilerConstant(CALLEE); - final VarNode selfInit = + VarNode selfInit = new VarNode( newFunctionNode.getSource(), newFunctionNode.getToken(), @@ -420,7 +444,6 @@ final class Attr extends NodeOperatorVisitor { LOG.info("Accepting self symbol init ", selfInit, " for ", newFunctionNode.getName()); final List newStatements = new ArrayList<>(); - newStatements.add(selfInit); assert callee.getSymbol() != null && callee.getSymbol().hasSlot(); final IdentNode name = selfInit.getName(); @@ -428,9 +451,10 @@ final class Attr extends NodeOperatorVisitor { assert nameSymbol != null; - name.setSymbol(nameSymbol); - selfInit.setSymbol(nameSymbol); + selfInit = selfInit.setName((IdentNode)name.setSymbol(lc, nameSymbol)); + selfInit = (VarNode)selfInit.setSymbol(lc, nameSymbol); + newStatements.add(selfInit); newStatements.addAll(body.getStatements()); newFunctionNode = newFunctionNode.setBody(lc, body.setStatements(lc, newStatements)); } @@ -447,34 +471,32 @@ final class Attr extends NodeOperatorVisitor { end(newFunctionNode, false); - return newFunctionNode; //.setFlag(lc, lc.getFlags(functionNode)); + return newFunctionNode; } @Override public Node leaveCONVERT(final UnaryNode unaryNode) { assert false : "There should be no convert operators in IR during Attribution"; - end(unaryNode); - return unaryNode; + return end(unaryNode); } @Override - public boolean enterIdentNode(final IdentNode identNode) { + public Node leaveIdentNode(final IdentNode identNode) { final String name = identNode.getName(); start(identNode); + final LexicalContext lc = getLexicalContext(); + if (identNode.isPropertyName()) { // assign a pseudo symbol to property name final Symbol pseudoSymbol = pseudoSymbol(name); LOG.info("IdentNode is property name -> assigning pseudo symbol ", pseudoSymbol); LOG.unindent(); - identNode.setSymbol(pseudoSymbol); - return false; + return identNode.setSymbol(lc, pseudoSymbol); } - final LexicalContext lc = getLexicalContext(); - final Block block = lc.getCurrentBlock(); - final Symbol oldSymbol = identNode.getSymbol(); + final Block block = lc.getCurrentBlock(); Symbol symbol = findSymbol(block, name); @@ -495,12 +517,11 @@ final class Attr extends NodeOperatorVisitor { } } - identNode.setSymbol(symbol); // if symbol is non-local or we're in a with block, we need to put symbol in scope (if it isn't already) maybeForceScope(symbol); } else { LOG.info("No symbol exists. Declare undefined: ", symbol); - symbol = defineSymbol(block, name, IS_GLOBAL, identNode); + symbol = defineSymbol(block, name, IS_GLOBAL); // we have never seen this before, it can be undefined newType(symbol, Type.OBJECT); // TODO unknown -we have explicit casts anyway? symbol.setCanBeUndefined(); @@ -509,14 +530,14 @@ final class Attr extends NodeOperatorVisitor { setBlockScope(name, symbol); - if (symbol != oldSymbol && !identNode.isInitializedHere()) { + if (symbol != null && !identNode.isInitializedHere()) { symbol.increaseUseCount(); } addLocalUse(identNode.getName()); end(identNode); - return false; + return identNode.setSymbol(lc, symbol); } /** @@ -525,7 +546,7 @@ final class Attr extends NodeOperatorVisitor { * @param symbol the symbol that might be scoped */ private void maybeForceScope(final Symbol symbol) { - if(!symbol.isScope() && symbolNeedsToBeScope(symbol)) { + if (!symbol.isScope() && symbolNeedsToBeScope(symbol)) { Symbol.setSymbolIsScope(getLexicalContext(), symbol); } } @@ -612,11 +633,11 @@ final class Attr extends NodeOperatorVisitor { private Symbol findSymbol(final Block block, final String name) { // Search up block chain to locate symbol. - for(final Iterator blocks = getLexicalContext().getBlocks(block); blocks.hasNext();) { + for (final Iterator blocks = getLexicalContext().getBlocks(block); blocks.hasNext();) { // Find name. final Symbol symbol = blocks.next().getExistingSymbol(name); // If found then we are good. - if(symbol != null) { + if (symbol != null) { return symbol; } } @@ -625,39 +646,19 @@ final class Attr extends NodeOperatorVisitor { @Override public Node leaveIndexNode(final IndexNode indexNode) { - ensureSymbol(Type.OBJECT, indexNode); //TODO - return indexNode; + return end(ensureSymbol(Type.OBJECT, indexNode)); } @SuppressWarnings("rawtypes") @Override - public boolean enterLiteralNode(final LiteralNode literalNode) { - try { - start(literalNode); - assert !literalNode.isTokenType(TokenType.THIS) : "tokentype for " + literalNode + " is this"; //guard against old dead code case. literal nodes should never inherit tokens - - if (literalNode instanceof ArrayLiteralNode) { - final ArrayLiteralNode arrayLiteralNode = (ArrayLiteralNode)literalNode; - final Node[] array = arrayLiteralNode.getValue(); - - for (int i = 0; i < array.length; i++) { - final Node element = array[i]; - if (element != null) { - array[i] = element.accept(this); - } - } - arrayLiteralNode.analyze(); - //array literal node now has an element type and all elements are attributed - } else { - assert !(literalNode.getValue() instanceof Node) : "literals with Node values not supported"; - } - - getLexicalContext().getCurrentFunction().newLiteral(literalNode); - } finally { - end(literalNode); + public Node leaveLiteralNode(final LiteralNode literalNode) { + assert !literalNode.isTokenType(TokenType.THIS) : "tokentype for " + literalNode + " is this"; //guard against old dead code case. literal nodes should never inherit tokens + assert literalNode instanceof ArrayLiteralNode || !(literalNode.getValue() instanceof Node) : "literals with Node values not supported"; + final Symbol symbol = new Symbol(getLexicalContext().getCurrentFunction().uniqueName(LITERAL_PREFIX.symbolName()), IS_CONSTANT, literalNode.getType()); + if (literalNode instanceof ArrayLiteralNode) { + ((ArrayLiteralNode)literalNode).analyze(); } - - return false; + return literalNode.setSymbol(getLexicalContext(), symbol); } @Override @@ -667,18 +668,13 @@ final class Attr extends NodeOperatorVisitor { @Override public Node leaveObjectNode(final ObjectNode objectNode) { - ensureSymbol(Type.OBJECT, objectNode); - return end(objectNode); + return end(ensureSymbol(Type.OBJECT, objectNode)); } - //TODO is this correct why not leave? @Override - public boolean enterPropertyNode(final PropertyNode propertyNode) { + public Node leavePropertyNode(final PropertyNode propertyNode) { // assign a pseudo symbol to property name, see NASHORN-710 - start(propertyNode); - propertyNode.setSymbol(new Symbol(propertyNode.getKeyName(), 0, Type.OBJECT)); - end(propertyNode); - return true; + return propertyNode.setSymbol(getLexicalContext(), new Symbol(propertyNode.getKeyName(), 0, Type.OBJECT)); } @Override @@ -763,12 +759,9 @@ final class Attr extends NodeOperatorVisitor { final IdentNode ident = varNode.getName(); final String name = ident.getName(); - final Symbol symbol = defineSymbol(getLexicalContext().getCurrentBlock(), name, IS_VAR, ident); + final Symbol symbol = defineSymbol(getLexicalContext().getCurrentBlock(), name, IS_VAR); assert symbol != null; - LOG.info("VarNode ", varNode, " set symbol ", symbol); - varNode.setSymbol(symbol); - // NASHORN-467 - use before definition of vars - conservative if (isLocalUse(ident.getName())) { newType(symbol, Type.OBJECT); @@ -780,22 +773,31 @@ final class Attr extends NodeOperatorVisitor { @Override public Node leaveVarNode(final VarNode varNode) { - final Node init = varNode.getInit(); - final IdentNode ident = varNode.getName(); + VarNode newVarNode = varNode; + + final Node init = newVarNode.getInit(); + final IdentNode ident = newVarNode.getName(); final String name = ident.getName(); if (init == null) { // var x; with no init will be treated like a use of x by - // visit(IdentNode) unless we remove the name - // from the localdef list. + // leaveIdentNode unless we remove the name from the localdef list. removeLocalDef(name); - return varNode; + return newVarNode; } addLocalDef(name); - final Symbol symbol = varNode.getSymbol(); - final boolean isScript = getLexicalContext().getDefiningFunction(symbol).isProgram(); //see NASHORN-56 + final LexicalContext lc = getLexicalContext(); + final Symbol symbol = findSymbol(lc.getCurrentBlock(), ident.getName()); + assert symbol != null; + + final IdentNode newIdent = (IdentNode)ident.setSymbol(lc, symbol); + + newVarNode = newVarNode.setName(newIdent); + newVarNode = (VarNode)newVarNode.setSymbol(lc, symbol); + + final boolean isScript = lc.getDefiningFunction(symbol).isProgram(); //see NASHORN-56 if ((init.getType().isNumeric() || init.getType().isBoolean()) && !isScript) { // Forbid integers as local vars for now as we have no way to treat them as undefined newType(symbol, init.getType()); @@ -803,25 +805,19 @@ final class Attr extends NodeOperatorVisitor { newType(symbol, Type.OBJECT); } - assert varNode.hasType() : varNode; + assert newVarNode.hasType() : newVarNode + " has no type"; - end(varNode); - - return varNode; + return end(newVarNode); } @Override public Node leaveADD(final UnaryNode unaryNode) { - ensureSymbol(arithType(), unaryNode); - end(unaryNode); - return unaryNode; + return end(ensureSymbol(arithType(), unaryNode)); } @Override public Node leaveBIT_NOT(final UnaryNode unaryNode) { - ensureSymbol(Type.INT, unaryNode); - end(unaryNode); - return unaryNode; + return end(ensureSymbol(Type.INT, unaryNode)); } @Override @@ -830,9 +826,7 @@ final class Attr extends NodeOperatorVisitor { ensureAssignmentSlots(getLexicalContext().getCurrentFunction(), unaryNode.rhs()); final Type type = arithType(); newType(unaryNode.rhs().getSymbol(), type); - ensureSymbol(type, unaryNode); - end(unaryNode); - return unaryNode; + return end(ensureSymbol(type, unaryNode)); } @Override @@ -908,23 +902,25 @@ final class Attr extends NodeOperatorVisitor { @Override public Node leaveNEW(final UnaryNode unaryNode) { - ensureSymbol(Type.OBJECT, unaryNode); - end(unaryNode); - return unaryNode; + return end(ensureSymbol(Type.OBJECT, unaryNode)); } @Override public Node leaveNOT(final UnaryNode unaryNode) { - ensureSymbol(Type.BOOLEAN, unaryNode); - end(unaryNode); - return unaryNode; + return end(ensureSymbol(Type.BOOLEAN, unaryNode)); } private IdentNode compilerConstant(CompilerConstants cc) { final FunctionNode functionNode = getLexicalContext().getCurrentFunction(); - final IdentNode node = new IdentNode(functionNode.getSource(), functionNode.getToken(), functionNode.getFinish(), cc.symbolName()); - node.setSymbol(functionNode.compilerConstant(cc)); - return node; + return (IdentNode) + new IdentNode( + functionNode.getSource(), + functionNode.getToken(), + functionNode.getFinish(), + cc.symbolName()). + setSymbol( + getLexicalContext(), + functionNode.compilerConstant(cc)); } @Override @@ -952,15 +948,12 @@ final class Attr extends NodeOperatorVisitor { @Override public Node leaveRuntimeNode(final RuntimeNode runtimeNode) { - ensureSymbol(runtimeNode.getRequest().getReturnType(), runtimeNode); - return runtimeNode; + return end(ensureSymbol(runtimeNode.getRequest().getReturnType(), runtimeNode)); } @Override public Node leaveSUB(final UnaryNode unaryNode) { - ensureSymbol(arithType(), unaryNode); - end(unaryNode); - return unaryNode; + return end(ensureSymbol(arithType(), unaryNode)); } @Override @@ -982,18 +975,16 @@ final class Attr extends NodeOperatorVisitor { ensureTypeNotUnknown(lhs); ensureTypeNotUnknown(rhs); - ensureSymbol(Type.widest(lhs.getType(), rhs.getType()), binaryNode); - - end(binaryNode); - - return binaryNode; + //even if we are adding two known types, this can overflow. i.e. + //int and number -> number. + //int and int are also number though. + //something and object is object + return end(ensureSymbol(Type.widest(arithType(), Type.widest(lhs.getType(), rhs.getType())), binaryNode)); } @Override public Node leaveAND(final BinaryNode binaryNode) { - ensureSymbol(Type.OBJECT, binaryNode); - end(binaryNode); - return binaryNode; + return end(ensureSymbol(Type.OBJECT, binaryNode)); } /** @@ -1013,8 +1004,7 @@ final class Attr extends NodeOperatorVisitor { Symbol symbol = findSymbol(block, name); if (symbol == null) { - symbol = defineSymbol(block, name, IS_GLOBAL, ident); - binaryNode.setSymbol(symbol); + symbol = defineSymbol(block, name, IS_GLOBAL); } else { maybeForceScope(symbol); } @@ -1025,6 +1015,31 @@ final class Attr extends NodeOperatorVisitor { return true; } + + /** + * This assign helper is called after an assignment, when all children of + * the assign has been processed. It fixes the types and recursively makes + * sure that everyhing has slots that should have them in the chain. + * + * @param binaryNode assignment node + */ + private Node leaveAssignmentNode(final BinaryNode binaryNode) { + BinaryNode newBinaryNode = binaryNode; + + final Node lhs = binaryNode.lhs(); + final Node rhs = binaryNode.rhs(); + final Type type; + + if (rhs.getType().isNumeric()) { + type = Type.widest(binaryNode.lhs().getType(), binaryNode.rhs().getType()); + } else { + type = Type.OBJECT; //force lhs to be an object if not numeric assignment, e.g. strings too. + } + + newType(lhs.getSymbol(), type); + return end(ensureSymbol(type, newBinaryNode)); + } + private boolean isLocal(FunctionNode function, Symbol symbol) { final FunctionNode definingFn = getLexicalContext().getDefiningFunction(symbol); // Temp symbols are not assigned to a block, so their defining fn is null; those can be assumed local @@ -1173,14 +1188,12 @@ final class Attr extends NodeOperatorVisitor { @Override public Node leaveCOMMARIGHT(final BinaryNode binaryNode) { - ensureSymbol(binaryNode.rhs().getType(), binaryNode); - return binaryNode; + return end(ensureSymbol(binaryNode.rhs().getType(), binaryNode)); } @Override public Node leaveCOMMALEFT(final BinaryNode binaryNode) { - ensureSymbol(binaryNode.lhs().getType(), binaryNode); - return binaryNode; + return end(ensureSymbol(binaryNode.lhs().getType(), binaryNode)); } @Override @@ -1189,15 +1202,10 @@ final class Attr extends NodeOperatorVisitor { } private Node leaveCmp(final BinaryNode binaryNode) { - final Node lhs = binaryNode.lhs(); - final Node rhs = binaryNode.rhs(); + ensureTypeNotUnknown(binaryNode.lhs()); + ensureTypeNotUnknown(binaryNode.rhs()); - ensureSymbol(Type.BOOLEAN, binaryNode); - ensureTypeNotUnknown(lhs); - ensureTypeNotUnknown(rhs); - - end(binaryNode); - return binaryNode; + return end(ensureSymbol(Type.BOOLEAN, binaryNode)); } private Node coerce(final BinaryNode binaryNode, final Type operandType, final Type destType) { @@ -1207,11 +1215,9 @@ final class Attr extends NodeOperatorVisitor { // as, say, an int : function(x) { return x & 4711 }, and x is not defined in // the function. to make this work, uncomment the following two type inferences // and debug. - //newType(binaryNode.lhs().getSymbol(), operandType); //newType(binaryNode.rhs().getSymbol(), operandType); - ensureSymbol(destType, binaryNode); - return binaryNode; + return ensureSymbol(destType, binaryNode); } private Node coerce(final BinaryNode binaryNode, final Type type) { @@ -1295,9 +1301,7 @@ final class Attr extends NodeOperatorVisitor { @Override public Node leaveOR(final BinaryNode binaryNode) { - ensureSymbol(Type.OBJECT, binaryNode); - end(binaryNode); - return binaryNode; + return end(ensureSymbol(Type.OBJECT, binaryNode)); } @Override @@ -1346,50 +1350,13 @@ final class Attr extends NodeOperatorVisitor { ensureTypeNotUnknown(rhs); final Type type = Type.widest(lhs.getType(), rhs.getType()); - ensureSymbol(type, ternaryNode); - - end(ternaryNode); - assert ternaryNode.getSymbol() != null; - - return ternaryNode; + return end(ensureSymbol(type, ternaryNode)); } - private void initThis(final Block block) { - final Symbol thisSymbol = defineSymbol(block, THIS.symbolName(), IS_PARAM | IS_THIS, null); - newType(thisSymbol, Type.OBJECT); - thisSymbol.setNeedsSlot(true); - } - - private void initScope(final Block block) { - final Symbol scopeSymbol = defineSymbol(block, SCOPE.symbolName(), IS_VAR | IS_INTERNAL, null); - newType(scopeSymbol, Type.typeFor(ScriptObject.class)); - scopeSymbol.setNeedsSlot(true); - } - - private void initReturn(final Block block) { - final Symbol returnSymbol = defineSymbol(block, RETURN.symbolName(), IS_VAR | IS_INTERNAL, null); - newType(returnSymbol, Type.OBJECT); - returnSymbol.setNeedsSlot(true); - //return symbol is always object as it's the __return__ thing. What returnType is is another matter though - } - - private void initVarArg(final Block block, final boolean needsArguments) { - final Symbol varArgsSymbol = defineSymbol(block, VARARGS.symbolName(), IS_PARAM | IS_INTERNAL, null); - varArgsSymbol.setTypeOverride(Type.OBJECT_ARRAY); - varArgsSymbol.setNeedsSlot(true); - - if (needsArguments) { - final Symbol argumentsSymbol = defineSymbol(block, ARGUMENTS.symbolName(), IS_VAR | IS_INTERNAL, null); - newType(argumentsSymbol, Type.typeFor(ScriptObject.class)); - argumentsSymbol.setNeedsSlot(true); - addLocalDef(ARGUMENTS.symbolName()); - } - } - - private void initCallee(final Block block) { - final Symbol calleeSymbol = defineSymbol(block, CALLEE.symbolName(), IS_PARAM | IS_INTERNAL, null); - newType(calleeSymbol, FunctionNode.FUNCTION_TYPE); - calleeSymbol.setNeedsSlot(true); + private void initCompileConstant(final CompilerConstants cc, final Block block, final int flags, final Type type) { + final Symbol symbol = defineSymbol(block, cc.symbolName(), flags); + newType(symbol, type); + symbol.setNeedsSlot(true); } /** @@ -1399,19 +1366,28 @@ final class Attr extends NodeOperatorVisitor { * @param functionNode the function node */ private void initParameters(final FunctionNode functionNode, final Block body) { + int pos = 0; for (final IdentNode param : functionNode.getParameters()) { addLocalDef(param.getName()); - final Symbol paramSymbol = defineSymbol(body, param.getName(), IS_PARAM, param); + + final Type callSiteParamType = functionNode.getHints().getParameterType(pos); + int flags = IS_PARAM; + if (callSiteParamType != null) { + LOG.info("Param ", param, " has a callsite type ", callSiteParamType, ". Using that."); + flags |= Symbol.IS_SPECIALIZED_PARAM; + } + + final Symbol paramSymbol = defineSymbol(body, param.getName(), flags); + assert paramSymbol != null; + if (paramSymbol != null) { - final Type callSiteParamType = functionNode.getSpecializedType(param); - if (callSiteParamType != null) { - LOG.info("Param ", paramSymbol, " has a callsite type ", callSiteParamType, ". Using that."); - } newType(paramSymbol, callSiteParamType == null ? Type.UNKNOWN : callSiteParamType); } - LOG.info("Initialized param ", paramSymbol); + LOG.info("Initialized param ", pos, "=", paramSymbol); + pos++; } + } /** @@ -1420,14 +1396,19 @@ final class Attr extends NodeOperatorVisitor { * * @param functionNode functionNode */ - private static void finalizeParameters(final FunctionNode functionNode) { + private FunctionNode finalizeParameters(final FunctionNode functionNode) { + final List newParams = new ArrayList<>(); final boolean isVarArg = functionNode.isVarArg(); - for (final IdentNode ident : functionNode.getParameters()) { - final Symbol paramSymbol = ident.getSymbol(); + int pos = 0; + for (final IdentNode param : functionNode.getParameters()) { + final Symbol paramSymbol = functionNode.getBody().getExistingSymbol(param.getName()); + assert paramSymbol != null; + assert paramSymbol.isParam(); + newParams.add((IdentNode)param.setSymbol(getLexicalContext(), paramSymbol)); assert paramSymbol != null; - Type type = functionNode.getSpecializedType(ident); + Type type = functionNode.getHints().getParameterType(pos); if (type == null) { type = Type.OBJECT; } @@ -1436,7 +1417,7 @@ final class Attr extends NodeOperatorVisitor { // this function, we can tell the runtime system that no matter what the // call site is, use this information. TODO if (!paramSymbol.getSymbolType().isObject()) { - LOG.finest("Parameter ", ident, " could profit from specialization to ", paramSymbol.getSymbolType()); + LOG.finest("Parameter ", param, " could profit from specialization to ", paramSymbol.getSymbolType()); } newType(paramSymbol, Type.widest(type, paramSymbol.getSymbolType())); @@ -1445,7 +1426,11 @@ final class Attr extends NodeOperatorVisitor { if (isVarArg) { paramSymbol.setNeedsSlot(false); } + + pos++; } + + return functionNode.setParameters(getLexicalContext(), newParams); } /** @@ -1459,7 +1444,7 @@ final class Attr extends NodeOperatorVisitor { for (final Property property : map.getProperties()) { final String key = property.getKey(); - final Symbol symbol = defineSymbol(block, key, IS_GLOBAL, null); + final Symbol symbol = defineSymbol(block, key, IS_GLOBAL); newType(symbol, Type.OBJECT); LOG.info("Added global symbol from property map ", symbol); } @@ -1498,7 +1483,7 @@ final class Attr extends NodeOperatorVisitor { * objects as parameters, for example +, but not *, which is known * to coerce types into doubles */ - if (node.getType().isUnknown() || symbol.isParam()) { + if (node.getType().isUnknown() || (symbol.isParam() && !symbol.isSpecializedParam())) { newType(symbol, Type.OBJECT); symbol.setCanBeUndefined(); } @@ -1614,29 +1599,6 @@ final class Attr extends NodeOperatorVisitor { } while (!changed.isEmpty()); } - /** - * This assign helper is called after an assignment, when all children of - * the assign has been processed. It fixes the types and recursively makes - * sure that everyhing has slots that should have them in the chain. - * - * @param binaryNode assignment node - */ - private Node leaveAssignmentNode(final BinaryNode binaryNode) { - final Node lhs = binaryNode.lhs(); - final Node rhs = binaryNode.rhs(); - - final Type type; - if (rhs.getType().isNumeric()) { - type = Type.widest(binaryNode.lhs().getType(), binaryNode.rhs().getType()); - } else { - type = Type.OBJECT; //force lhs to be an object if not numeric assignment, e.g. strings too. - } - ensureSymbol(type, binaryNode); - newType(lhs.getSymbol(), type); - end(binaryNode); - return binaryNode; - } - private Node leaveSelfModifyingAssignmentNode(final BinaryNode binaryNode) { return leaveSelfModifyingAssignmentNode(binaryNode, binaryNode.getWidestOperationType()); } @@ -1646,25 +1608,20 @@ final class Attr extends NodeOperatorVisitor { final Node lhs = binaryNode.lhs(); newType(lhs.getSymbol(), destType); //may not narrow if dest is already wider than destType - ensureSymbol(destType, binaryNode); //for OP= nodes, the node can carry a narrower types than its lhs rhs. This is perfectly fine +// ensureSymbol(destType, binaryNode); //for OP= nodes, the node can carry a narrower types than its lhs rhs. This is perfectly fine ensureAssignmentSlots(getLexicalContext().getCurrentFunction(), binaryNode); - end(binaryNode); - return binaryNode; + return end(ensureSymbol(destType, binaryNode)); } - private Symbol ensureSymbol(final FunctionNode functionNode, final Type type, final Node node) { - LOG.info("New TEMPORARY added to ", functionNode.getName(), " type=", type); - return functionNode.ensureSymbol(getLexicalContext().getCurrentBlock(), type, node); - } - - private Symbol ensureSymbol(final Type type, final Node node) { - return ensureSymbol(getLexicalContext().getCurrentFunction(), type, node); + private Node ensureSymbol(final Type type, final Node node) { + LOG.info("New TEMPORARY added to ", getLexicalContext().getCurrentFunction().getName(), " type=", type); + return Attr.ensureSymbol(getLexicalContext(), getLexicalContext().getCurrentBlock(), type, node); } private Symbol newInternal(final String name, final Type type) { - final Symbol iter = defineSymbol(getLexicalContext().getCurrentBlock(), name, IS_VAR | IS_INTERNAL, null); + final Symbol iter = defineSymbol(getLexicalContext().getCurrentBlock(), name, IS_VAR | IS_INTERNAL); iter.setType(type); // NASHORN-73 return iter; } @@ -1721,6 +1678,17 @@ final class Attr extends NodeOperatorVisitor { localUses.peek().add(name); } + static Node ensureSymbol(final LexicalContext lc, final Block block, final Type type, final Node node) { + Symbol symbol = node.getSymbol(); + if (symbol != null) { + return node; + } + final String uname = lc.getCurrentFunction().uniqueName(TEMP_PREFIX.symbolName()); + symbol = new Symbol(uname, IS_TEMP, type); + block.putSymbol(lc, symbol); + return node.setSymbol(lc, symbol); + } + /** * Pessimistically promote all symbols in current function node to Object types * This is done when the function contains unevaluated black boxes such as @@ -1731,8 +1699,7 @@ final class Attr extends NodeOperatorVisitor { private static void objectifySymbols(final Block body) { body.accept(new NodeVisitor() { private void toObject(final Block block) { - for (final Iterator iter = block.symbolIterator(); iter.hasNext();) { - final Symbol symbol = iter.next(); + for (final Symbol symbol : block.getSymbols()) { if (!symbol.isTemp()) { newType(symbol, Type.OBJECT); } diff --git a/nashorn/src/jdk/nashorn/internal/codegen/CodeGenerator.java b/nashorn/src/jdk/nashorn/internal/codegen/CodeGenerator.java index 5d4b8f9846c..8f5b3154d25 100644 --- a/nashorn/src/jdk/nashorn/internal/codegen/CodeGenerator.java +++ b/nashorn/src/jdk/nashorn/internal/codegen/CodeGenerator.java @@ -568,8 +568,7 @@ final class CodeGenerator extends NodeOperatorVisitor { * @param block block containing symbols. */ private void symbolInfo(final Block block) { - for (final Iterator iter = block.symbolIterator(); iter.hasNext(); ) { - final Symbol symbol = iter.next(); + for (final Symbol symbol : block.getSymbols()) { if (symbol.hasSlot()) { method.localVariable(symbol, block.getEntryLabel(), block.getBreakLabel()); } @@ -937,11 +936,10 @@ final class CodeGenerator extends NodeOperatorVisitor { private static int assignSlots(final Block block, final int firstSlot) { int nextSlot = firstSlot; - for (final Iterator iter = block.symbolIterator(); iter.hasNext(); ) { - final Symbol next = iter.next(); - if (next.hasSlot()) { - next.setSlot(nextSlot); - nextSlot += next.slotCount(); + for (final Symbol symbol : block.getSymbols()) { + if (symbol.hasSlot()) { + symbol.setSlot(nextSlot); + nextSlot += symbol.slotCount(); } } return nextSlot; @@ -1002,10 +1000,7 @@ final class CodeGenerator extends NodeOperatorVisitor { final boolean hasArguments = function.needsArguments(); - final Iterator symbols = block.symbolIterator(); - - while (symbols.hasNext()) { - final Symbol symbol = symbols.next(); + for (final Symbol symbol : block.getSymbols()) { if (symbol.isInternal() || symbol.isThis() || symbol.isTemp()) { continue; @@ -1076,12 +1071,7 @@ final class CodeGenerator extends NodeOperatorVisitor { } } - final Iterator iter = block.symbolIterator(); - final List symbols = new ArrayList<>(); - while (iter.hasNext()) { - symbols.add(iter.next()); - } - initSymbols(symbols); + initSymbols(block.getSymbols()); } // Debugging: print symbols? @see --print-symbols flag @@ -2364,7 +2354,6 @@ final class CodeGenerator extends NodeOperatorVisitor { public boolean enterDISCARD(final UnaryNode unaryNode) { final Node rhs = unaryNode.rhs(); - // System.err.println("**** Enter discard " + unaryNode); discard.push(rhs); load(rhs); @@ -2373,7 +2362,7 @@ final class CodeGenerator extends NodeOperatorVisitor { method.pop(); discard.pop(); } - // System.err.println("**** Leave discard " + unaryNode); + return false; } diff --git a/nashorn/src/jdk/nashorn/internal/codegen/CompilationPhase.java b/nashorn/src/jdk/nashorn/internal/codegen/CompilationPhase.java index 884a68bfc2f..22966307a25 100644 --- a/nashorn/src/jdk/nashorn/internal/codegen/CompilationPhase.java +++ b/nashorn/src/jdk/nashorn/internal/codegen/CompilationPhase.java @@ -42,7 +42,7 @@ enum CompilationPhase { */ LAZY_INITIALIZATION_PHASE(EnumSet.of(INITIALIZED, PARSED)) { @Override - FunctionNode transform(final Compiler compiler, final FunctionNode fn0) { + FunctionNode transform(final Compiler compiler, final FunctionNode fn) { /* * For lazy compilation, we might be given a node previously marked @@ -58,8 +58,7 @@ enum CompilationPhase { * function from a trampoline */ - final FunctionNode outermostFunctionNode = compiler.getFunctionNode(); - assert outermostFunctionNode == fn0; + final FunctionNode outermostFunctionNode = fn; final Set neverLazy = new HashSet<>(); final Set lazy = new HashSet<>(); @@ -172,20 +171,26 @@ enum CompilationPhase { ATTRIBUTION_PHASE(EnumSet.of(INITIALIZED, PARSED, CONSTANT_FOLDED, LOWERED)) { @Override FunctionNode transform(final Compiler compiler, final FunctionNode fn) { - return (FunctionNode)initReturnTypes(fn).accept(new Attr()); + return (FunctionNode)enterAttr(fn).accept(new Attr()); } /** * Pessimistically set all lazy functions' return types to Object + * and the function symbols to object * @param functionNode node where to start iterating */ - private FunctionNode initReturnTypes(final FunctionNode functionNode) { + private FunctionNode enterAttr(final FunctionNode functionNode) { return (FunctionNode)functionNode.accept(new NodeVisitor() { @Override public Node leaveFunctionNode(final FunctionNode node) { - return node.isLazy() ? - node.setReturnType(getLexicalContext(), Type.OBJECT) : - node.setReturnType(getLexicalContext(), Type.UNKNOWN); + final LexicalContext lc = getLexicalContext(); + if (node.isLazy()) { + FunctionNode newNode = node.setReturnType(getLexicalContext(), Type.OBJECT); + return Attr.ensureSymbol(lc, lc.getCurrentBlock(), Type.OBJECT, newNode); + } + //node may have a reference here that needs to be nulled if it was referred to by + //its outer context, if it is lazy and not attributed + return node.setReturnType(lc, Type.UNKNOWN).setSymbol(lc, null); } }); } @@ -207,6 +212,7 @@ enum CompilationPhase { FunctionNode transform(final Compiler compiler, final FunctionNode fn) { final CompileUnit outermostCompileUnit = compiler.addCompileUnit(compiler.firstCompileUnitName()); +// assert fn.isProgram() ; final FunctionNode newFunctionNode = new Splitter(compiler, fn, outermostCompileUnit).split(fn); assert newFunctionNode.getCompileUnit() == outermostCompileUnit : "fn.compileUnit (" + newFunctionNode.getCompileUnit() + ") != " + outermostCompileUnit; @@ -216,15 +222,6 @@ enum CompilationPhase { compiler.setStrictMode(true); } - /* - newFunctionNode.accept(new NodeVisitor() { - @Override - public boolean enterFunctionNode(final FunctionNode functionNode) { - assert functionNode.getCompileUnit() != null : functionNode.getName() + " " + Debug.id(functionNode) + " has no compile unit"; - return true; - } - });*/ - return newFunctionNode; } diff --git a/nashorn/src/jdk/nashorn/internal/codegen/Compiler.java b/nashorn/src/jdk/nashorn/internal/codegen/Compiler.java index b1e47d0293d..2f0c0d2272b 100644 --- a/nashorn/src/jdk/nashorn/internal/codegen/Compiler.java +++ b/nashorn/src/jdk/nashorn/internal/codegen/Compiler.java @@ -77,6 +77,8 @@ public final class Compiler { /** Name of the objects package */ public static final String OBJECTS_PACKAGE = "jdk/nashorn/internal/objects"; + private Source source; + private final Map bytecode; private final Set compileUnits; @@ -87,12 +89,10 @@ public final class Compiler { private final ScriptEnvironment env; - private final String scriptName; + private String scriptName; private boolean strict; - private FunctionNode functionNode; - private CodeInstaller installer; /** logger for compiler, trampolines, splits and related code generation events @@ -167,6 +167,41 @@ public final class Compiler { } } + /** + * Environment information known to the compile, e.g. params + */ + public static class Hints { + private final Type[] paramTypes; + + /** singleton empty hints */ + public static final Hints EMPTY = new Hints(); + + private Hints() { + this.paramTypes = null; + } + + /** + * Constructor + * @param paramTypes known parameter types for this callsite + */ + public Hints(final Type[] paramTypes) { + this.paramTypes = paramTypes; + } + + /** + * Get the parameter type for this parameter position, or + * null if now known + * @param pos position + * @return parameter type for this callsite if known + */ + public Type getParameterType(final int pos) { + if (paramTypes != null && pos < paramTypes.length) { + return paramTypes[pos]; + } + return null; + } + } + /** * Standard (non-lazy) compilation, that basically will take an entire script * and JIT it at once. This can lead to long startup time and fewer type @@ -207,21 +242,22 @@ public final class Compiler { * @param strict should this compilation use strict mode semantics */ //TODO support an array of FunctionNodes for batch lazy compilation - Compiler(final ScriptEnvironment env, final CodeInstaller installer, final FunctionNode functionNode, final CompilationSequence sequence, final boolean strict) { + Compiler(final ScriptEnvironment env, final CodeInstaller installer, final CompilationSequence sequence, final boolean strict) { this.env = env; - this.functionNode = functionNode; this.sequence = sequence; this.installer = installer; - this.strict = strict || functionNode.isStrict(); this.constantData = new ConstantData(); this.compileUnits = new HashSet<>(); this.bytecode = new HashMap<>(); + } + private void initCompiler(final FunctionNode functionNode) { + this.strict = strict || functionNode.isStrict(); final StringBuilder sb = new StringBuilder(); sb.append(functionNode.uniqueName(DEFAULT_SCRIPT_NAME.symbolName() + lazyTag(functionNode))). append('$'). append(safeSourceName(functionNode.getSource())); - + this.source = functionNode.getSource(); this.scriptName = sb.toString(); } @@ -229,52 +265,43 @@ public final class Compiler { * Constructor * * @param installer code installer - * @param functionNode function node (in any available {@link CompilationState}) to compile * @param strict should this compilation use strict mode semantics */ - public Compiler(final CodeInstaller installer, final FunctionNode functionNode, final boolean strict) { - this(installer.getOwner(), installer, functionNode, sequence(installer.getOwner()._lazy_compilation), strict); + public Compiler(final CodeInstaller installer, final boolean strict) { + this(installer.getOwner(), installer, sequence(installer.getOwner()._lazy_compilation), strict); } /** * Constructor - compilation will use the same strict semantics as in script environment * * @param installer code installer - * @param functionNode function node (in any available {@link CompilationState}) to compile */ - public Compiler(final CodeInstaller installer, final FunctionNode functionNode) { - this(installer.getOwner(), installer, functionNode, sequence(installer.getOwner()._lazy_compilation), installer.getOwner()._strict); + public Compiler(final CodeInstaller installer) { + this(installer.getOwner(), installer, sequence(installer.getOwner()._lazy_compilation), installer.getOwner()._strict); } /** * Constructor - compilation needs no installer, but uses a script environment * Used in "compile only" scenarios * @param env a script environment - * @param functionNode functionNode to compile */ - public Compiler(final ScriptEnvironment env, final FunctionNode functionNode) { - this(env, null, functionNode, sequence(env._lazy_compilation), env._strict); + public Compiler(final ScriptEnvironment env) { + this(env, null, sequence(env._lazy_compilation), env._strict); } /** * Execute the compilation this Compiler was created with - * @params param types if known, for specialization + * @param functionNode function node to compile from its current state * @throws CompilationException if something goes wrong * @return function node that results from code transforms */ - public FunctionNode compile() throws CompilationException { - return compile(null); - } + public FunctionNode compile(final FunctionNode functionNode) throws CompilationException { + FunctionNode newFunctionNode = functionNode; + + initCompiler(newFunctionNode); //TODO move this state into functionnode? - /** - * Execute the compilation this Compiler was created with - * @param paramTypes param types if known, for specialization - * @throws CompilationException if something goes wrong - * @return function node that results from code transforms - */ - public FunctionNode compile(final Class paramTypes) throws CompilationException { for (final String reservedName : RESERVED_NAMES) { - functionNode.uniqueName(reservedName); + newFunctionNode.uniqueName(reservedName); } final boolean fine = !LOG.levelAbove(Level.FINE); @@ -283,7 +310,7 @@ public final class Compiler { long time = 0L; for (final CompilationPhase phase : sequence) { - this.functionNode = phase.apply(this, functionNode); + newFunctionNode = phase.apply(this, newFunctionNode); final long duration = Timing.isEnabled() ? (phase.getEndTime() - phase.getStartTime()) : 0L; time += duration; @@ -293,7 +320,7 @@ public final class Compiler { sb.append(phase.toString()). append(" done for function '"). - append(functionNode.getName()). + append(newFunctionNode.getName()). append('\''); if (duration > 0L) { @@ -309,7 +336,7 @@ public final class Compiler { if (info) { final StringBuilder sb = new StringBuilder(); sb.append("Compile job for '"). - append(functionNode.getName()). + append(newFunctionNode.getName()). append("' finished"); if (time > 0L) { @@ -321,16 +348,15 @@ public final class Compiler { LOG.info(sb); } - return functionNode; + return newFunctionNode; } - private Class install(final String className, final byte[] code) { + private Class install(final FunctionNode functionNode, final String className, final byte[] code) { LOG.fine("Installing class ", className); final Class clazz = installer.install(Compiler.binaryName(className), code); try { - final Source source = getSource(); final Object[] constants = getConstantData().toArray(); // Need doPrivileged because these fields are private AccessController.doPrivileged(new PrivilegedExceptionAction() { @@ -355,9 +381,10 @@ public final class Compiler { /** * Install compiled classes into a given loader + * @param functionNode function node to install - must be in {@link CompilationState#EMITTED} state * @return root script class - if there are several compile units they will also be installed */ - public Class install() { + public Class install(final FunctionNode functionNode) { final long t0 = Timing.isEnabled() ? System.currentTimeMillis() : 0L; assert functionNode.hasState(CompilationState.EMITTED) : functionNode.getName() + " has no bytecode and cannot be installed"; @@ -366,7 +393,7 @@ public final class Compiler { final String rootClassName = firstCompileUnitName(); final byte[] rootByteCode = bytecode.get(rootClassName); - final Class rootClass = install(rootClassName, rootByteCode); + final Class rootClass = install(functionNode, rootClassName, rootByteCode); int length = rootByteCode.length; @@ -380,7 +407,7 @@ public final class Compiler { final byte[] code = entry.getValue(); length += code.length; - installedClasses.put(className, install(className, code)); + installedClasses.put(className, install(functionNode, className, code)); } for (final CompileUnit unit : compileUnits) { @@ -430,10 +457,6 @@ public final class Compiler { this.strict = strict; } - FunctionNode getFunctionNode() { - return functionNode; - } - ConstantData getConstantData() { return constantData; } @@ -442,10 +465,6 @@ public final class Compiler { return installer; } - Source getSource() { - return functionNode.getSource(); - } - void addClass(final String name, final byte[] code) { bytecode.put(name, code); } @@ -496,7 +515,7 @@ public final class Compiler { } private CompileUnit initCompileUnit(final String unitClassName, final long initialWeight) { - final ClassEmitter classEmitter = new ClassEmitter(env, functionNode.getSource().getName(), unitClassName, strict); + final ClassEmitter classEmitter = new ClassEmitter(env, source.getName(), unitClassName, strict); final CompileUnit compileUnit = new CompileUnit(unitClassName, classEmitter, initialWeight); classEmitter.begin(); diff --git a/nashorn/src/jdk/nashorn/internal/codegen/FinalizeTypes.java b/nashorn/src/jdk/nashorn/internal/codegen/FinalizeTypes.java index 5f6a63dc5ae..895b344e3e8 100644 --- a/nashorn/src/jdk/nashorn/internal/codegen/FinalizeTypes.java +++ b/nashorn/src/jdk/nashorn/internal/codegen/FinalizeTypes.java @@ -30,7 +30,6 @@ import static jdk.nashorn.internal.codegen.CompilerConstants.SCOPE; import java.util.ArrayList; import java.util.HashSet; -import java.util.Iterator; import java.util.List; import jdk.nashorn.internal.codegen.types.Type; import jdk.nashorn.internal.ir.AccessNode; @@ -354,13 +353,6 @@ final class FinalizeTypes extends NodeOperatorVisitor { return true; } - /* - @Override - public Node leaveBlock(final Block block) { - final LexicalContext lc = getLexicalContext(); - return block;//.setFlag(lc, lc.getFlags(block)); - }*/ - @Override public Node leaveCatchNode(final CatchNode catchNode) { final Node exceptionCondition = catchNode.getExceptionCondition(); @@ -551,8 +543,7 @@ final class FinalizeTypes extends NodeOperatorVisitor { final boolean allVarsInScope = functionNode.allVarsInScope(); final boolean isVarArg = functionNode.isVarArg(); - for (final Iterator iter = block.symbolIterator(); iter.hasNext(); ) { - final Symbol symbol = iter.next(); + for (final Symbol symbol : block.getSymbols()) { if (symbol.isInternal() || symbol.isThis() || symbol.isTemp()) { continue; } @@ -812,14 +803,12 @@ final class FinalizeTypes extends NodeOperatorVisitor { LOG.info("CONVERT('", node, "', ", to, ") => '", resultNode, "'"); + assert !node.isTerminal(); + final LexicalContext lc = getLexicalContext(); //This is the only place in this file that can create new temporaries //FinalizeTypes may not introduce ANY node that is not a conversion. - lc.getCurrentFunction().ensureSymbol(lc.getCurrentBlock(), to, resultNode); - - assert !node.isTerminal(); - - return resultNode; + return Attr.ensureSymbol(lc, lc.getCurrentBlock(), to, resultNode); } private static Node discard(final Node node) { @@ -905,7 +894,7 @@ final class FinalizeTypes extends NodeOperatorVisitor { if (literalNode != null) { //inherit literal symbol for attr. - literalNode.setSymbol(parent.getSymbol()); + literalNode = (LiteralNode)literalNode.setSymbol(null, parent.getSymbol()); } return literalNode; diff --git a/nashorn/src/jdk/nashorn/internal/codegen/ObjectCreator.java b/nashorn/src/jdk/nashorn/internal/codegen/ObjectCreator.java index 355dc6fdafb..b0e5a7730e9 100644 --- a/nashorn/src/jdk/nashorn/internal/codegen/ObjectCreator.java +++ b/nashorn/src/jdk/nashorn/internal/codegen/ObjectCreator.java @@ -27,7 +27,6 @@ package jdk.nashorn.internal.codegen; import java.util.List; import static jdk.nashorn.internal.codegen.ObjectClassGenerator.FIELD_PADDING; -import static jdk.nashorn.internal.codegen.ObjectClassGenerator.FIELD_ROUNDING; import jdk.nashorn.internal.ir.Symbol; import jdk.nashorn.internal.runtime.Context; import jdk.nashorn.internal.runtime.PropertyMap; diff --git a/nashorn/src/jdk/nashorn/internal/codegen/Splitter.java b/nashorn/src/jdk/nashorn/internal/codegen/Splitter.java index f482e4dfa33..8049473c87b 100644 --- a/nashorn/src/jdk/nashorn/internal/codegen/Splitter.java +++ b/nashorn/src/jdk/nashorn/internal/codegen/Splitter.java @@ -75,7 +75,7 @@ final class Splitter extends NodeVisitor { */ public Splitter(final Compiler compiler, final FunctionNode functionNode, final CompileUnit outermostCompileUnit) { this.compiler = compiler; - this.outermost = functionNode; + this.outermost = functionNode; this.outermostCompileUnit = outermostCompileUnit; } @@ -95,7 +95,7 @@ final class Splitter extends NodeVisitor { final LexicalContext lc = getLexicalContext(); long weight = WeighNodes.weigh(functionNode); - final boolean top = compiler.getFunctionNode() == outermost; + final boolean top = fn.isProgram(); //compiler.getFunctionNode() == outermost; if (weight >= SPLIT_THRESHOLD) { LOG.finest("Splitting '", functionNode.getName(), "' as its weight ", weight, " exceeds split threshold ", SPLIT_THRESHOLD); @@ -273,7 +273,9 @@ final class Splitter extends NodeVisitor { return literal; } - getLexicalContext().setFlag(getLexicalContext().getCurrentFunction(), FunctionNode.IS_SPLIT); + final FunctionNode functionNode = getLexicalContext().getCurrentFunction(); + + getLexicalContext().setFlag(functionNode, FunctionNode.IS_SPLIT); if (literal instanceof ArrayLiteralNode) { final ArrayLiteralNode arrayLiteralNode = (ArrayLiteralNode) literal; diff --git a/nashorn/src/jdk/nashorn/internal/ir/Block.java b/nashorn/src/jdk/nashorn/internal/ir/Block.java index 48b9ed69253..15a00869f0a 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/Block.java +++ b/nashorn/src/jdk/nashorn/internal/ir/Block.java @@ -30,7 +30,6 @@ import java.util.ArrayList; import java.util.Arrays; import java.util.Collections; import java.util.Comparator; -import java.util.Iterator; import java.util.LinkedHashMap; import java.util.List; import java.util.Map; @@ -104,18 +103,26 @@ public class Block extends BreakableNode implements Flags { this(source, token, finish, statements.toArray(new Node[statements.size()])); } - private Block(final Block block, final int finish, final List statements, final int flags) { + private Block(final Block block, final int finish, final List statements, final int flags, final Map symbols) { super(block); this.statements = statements; this.flags = flags; - this.symbols = block.symbols; //todo - symbols have no dependencies on any IR node and can as far as we understand it be shallow copied now + this.symbols = new LinkedHashMap<>(symbols); //todo - symbols have no dependencies on any IR node and can as far as we understand it be shallow copied now this.entryLabel = new Label(block.entryLabel); - this.finish = finish; + this.finish = finish; + } + + /** + * Clear the symbols in a block + * TODO: make this immutable + */ + public void clearSymbols() { + symbols.clear(); } @Override public Node ensureUniqueLabels(final LexicalContext lc) { - return Node.replaceInLexicalContext(lc, this, new Block(this, finish, statements, flags)); + return Node.replaceInLexicalContext(lc, this, new Block(this, finish, statements, flags, symbols)); } /** @@ -137,15 +144,15 @@ public class Block extends BreakableNode implements Flags { * Get an iterator for all the symbols defined in this block * @return symbol iterator */ - public Iterator symbolIterator() { - return symbols.values().iterator(); + public List getSymbols() { + return Collections.unmodifiableList(new ArrayList<>(symbols.values())); } /** * Retrieves an existing symbol defined in the current block. * @param name the name of the symbol * @return an existing symbol with the specified name defined in the current block, or null if this block doesn't - * define a symbol with this name. + * define a symbol with this name.T */ public Symbol getExistingSymbol(final String name) { return symbols.get(name); @@ -241,17 +248,17 @@ public class Block extends BreakableNode implements Flags { if (!statements.isEmpty()) { lastFinish = statements.get(statements.size() - 1).getFinish(); } - return Node.replaceInLexicalContext(lc, this, new Block(this, Math.max(finish, lastFinish), statements, flags)); + return Node.replaceInLexicalContext(lc, this, new Block(this, Math.max(finish, lastFinish), statements, flags, symbols)); } /** * Add or overwrite an existing symbol in the block * - * @param name name of symbol + * @param lc get lexical context * @param symbol symbol */ - public void putSymbol(final String name, final Symbol symbol) { - symbols.put(name, symbol); + public void putSymbol(final LexicalContext lc, final Symbol symbol) { + symbols.put(symbol.getName(), symbol); } /** @@ -268,7 +275,7 @@ public class Block extends BreakableNode implements Flags { if (this.flags == flags) { return this; } - return Node.replaceInLexicalContext(lc, this, new Block(this, finish, statements, flags)); + return Node.replaceInLexicalContext(lc, this, new Block(this, finish, statements, flags, symbols)); } @Override @@ -296,7 +303,7 @@ public class Block extends BreakableNode implements Flags { return this; } - return Node.replaceInLexicalContext(lc, this, new Block(this, finish, statements, flags | NEEDS_SCOPE)); + return Node.replaceInLexicalContext(lc, this, new Block(this, finish, statements, flags | NEEDS_SCOPE, symbols)); } /** @@ -306,13 +313,11 @@ public class Block extends BreakableNode implements Flags { * @return next slot */ public int nextSlot() { - final Iterator iter = symbolIterator(); int next = 0; - while (iter.hasNext()) { - final Symbol symbol = iter.next(); - if (symbol.hasSlot()) { - next += symbol.slotCount(); - } + for (final Symbol symbol : getSymbols()) { + if (symbol.hasSlot()) { + next += symbol.slotCount(); + } } return next; } diff --git a/nashorn/src/jdk/nashorn/internal/ir/CatchNode.java b/nashorn/src/jdk/nashorn/internal/ir/CatchNode.java index 5e1b4111de7..e7d231e8d82 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/CatchNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/CatchNode.java @@ -138,7 +138,12 @@ public final class CatchNode extends Node { return body; } - private CatchNode setException(final IdentNode exception) { + /** + * Resets the exception of a catch block + * @param exception new exception + * @return new catch node if changed, same otherwise + */ + public CatchNode setException(final IdentNode exception) { if (this.exception == exception) { return this; } diff --git a/nashorn/src/jdk/nashorn/internal/ir/FunctionNode.java b/nashorn/src/jdk/nashorn/internal/ir/FunctionNode.java index 60fe64ff7a4..dc11771de3b 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/FunctionNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/FunctionNode.java @@ -25,16 +25,12 @@ package jdk.nashorn.internal.ir; -import static jdk.nashorn.internal.codegen.CompilerConstants.LITERAL_PREFIX; -import static jdk.nashorn.internal.codegen.CompilerConstants.TEMP_PREFIX; -import static jdk.nashorn.internal.ir.Symbol.IS_CONSTANT; -import static jdk.nashorn.internal.ir.Symbol.IS_TEMP; - import java.util.Collections; import java.util.EnumSet; import java.util.HashSet; import java.util.List; import java.util.Set; + import jdk.nashorn.internal.codegen.CompileUnit; import jdk.nashorn.internal.codegen.Compiler; import jdk.nashorn.internal.codegen.CompilerConstants; @@ -95,6 +91,10 @@ public final class FunctionNode extends LexicalContextNode implements Flags compilationState; + @Ignore + private final Compiler.Hints hints; + /** Function flags. */ private final int flags; @@ -176,6 +179,9 @@ public final class FunctionNode extends LexicalContextNode implements Flags(); - this.flags = flags; - this.compileUnit = null; - this.body = null; + this.ident = ident; + this.name = name; + this.kind = kind; + this.parameters = parameters; + this.firstToken = firstToken; + this.lastToken = token; + this.namespace = namespace; + this.compilationState = EnumSet.of(CompilationState.INITIALIZED); + this.declaredSymbols = new HashSet<>(); + this.flags = flags; + this.compileUnit = null; + this.body = null; + this.snapshot = null; + this.hints = null; } - private FunctionNode(final FunctionNode functionNode, final long lastToken, final int flags, final Type returnType, final CompileUnit compileUnit, final EnumSet compilationState, final Block body) { + private FunctionNode( + final FunctionNode functionNode, + final long lastToken, + final int flags, + final Type returnType, + final CompileUnit compileUnit, + final EnumSet compilationState, + final Block body, + final List parameters, + final FunctionNode snapshot, + final Compiler.Hints hints) { super(functionNode); - this.flags = flags; - this.returnType = returnType; - this.compileUnit = compileUnit; - this.lastToken = lastToken; + + this.flags = flags; + this.returnType = returnType; + this.compileUnit = compileUnit; + this.lastToken = lastToken; this.compilationState = compilationState; - this.body = body; + this.body = body; + this.parameters = parameters; + this.snapshot = snapshot; + this.hints = hints; // the fields below never change - they are final and assigned in constructor - this.name = functionNode.name; - this.ident = functionNode.ident; - this.namespace = functionNode.namespace; + this.name = functionNode.name; + this.ident = functionNode.ident; + this.namespace = functionNode.namespace; this.declaredSymbols = functionNode.declaredSymbols; - this.kind = functionNode.kind; - this.parameters = functionNode.parameters; - this.firstToken = functionNode.firstToken; + this.kind = functionNode.kind; + this.firstToken = functionNode.firstToken; } @Override @@ -260,6 +281,36 @@ public final class FunctionNode extends LexicalContextNode implements Flags newState = EnumSet.copyOf(this.compilationState); newState.add(state); - return Node.replaceInLexicalContext(lc, this, new FunctionNode(this, lastToken, flags, returnType, compileUnit, newState, body)); + return Node.replaceInLexicalContext(lc, this, new FunctionNode(this, lastToken, flags, returnType, compileUnit, newState, body, parameters, snapshot, hints)); + } + + /** + * Get any compiler hints that may associated with the function + * @return compiler hints + */ + public Compiler.Hints getHints() { + return this.hints == null ? Compiler.Hints.EMPTY : hints; + } + + /** + * Set compiler hints for this function + * @param lc lexical context + * @param hints compiler hints + * @return new function if hints changed + */ + public FunctionNode setHints(final LexicalContext lc, final Compiler.Hints hints) { + if (this.hints == hints) { + return this; + } + return Node.replaceInLexicalContext(lc, this, new FunctionNode(this, lastToken, flags, returnType, compileUnit, compilationState, body, parameters, snapshot, hints)); } /** @@ -319,20 +391,6 @@ public final class FunctionNode extends LexicalContextNode implements Flags literalNode) { - final String uname = uniqueName(LITERAL_PREFIX.symbolName()); - final Symbol symbol = new Symbol(uname, IS_CONSTANT, literalNode.getType()); - literalNode.setSymbol(symbol); - - return symbol; - } @Override public void toString(final StringBuilder sb) { @@ -374,7 +432,7 @@ public final class FunctionNode extends LexicalContextNode implements Flags parameters) { + if (this.parameters == parameters) { + return this; + } + return Node.replaceInLexicalContext(lc, this, new FunctionNode(this, lastToken, flags, returnType, compileUnit, compilationState, body, parameters, snapshot, hints)); } /** @@ -674,7 +736,10 @@ public final class FunctionNode extends LexicalContextNode implements Flags= 0; i--) { if (stack[i] == node) { flags[i] |= flag; - //System.err.println("Setting flag " + node + " " + flag); return; } } @@ -117,8 +116,6 @@ public class LexicalContext { return (FunctionNode)stack[0]; } - - /** * Pushes a new block on top of the context, making it the innermost open block. * @param node the new node diff --git a/nashorn/src/jdk/nashorn/internal/ir/LexicalContextNode.java b/nashorn/src/jdk/nashorn/internal/ir/LexicalContextNode.java index e48c6e0b0d5..c4d78224271 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/LexicalContextNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/LexicalContextNode.java @@ -70,4 +70,16 @@ public abstract class LexicalContextNode extends Node { final LexicalContextNode newNode = (LexicalContextNode)accept(lc, visitor); return lc.pop(newNode); } + + /** + * Set the symbol and replace in lexical context if applicable + * @param lc lexical context + * @param symbol symbol + * @return new node if symbol changed + */ + @Override + public Node setSymbol(final LexicalContext lc, final Symbol symbol) { + return Node.replaceInLexicalContext(lc, this, (LexicalContextNode)super.setSymbol(null, symbol)); + } + } diff --git a/nashorn/src/jdk/nashorn/internal/ir/LiteralNode.java b/nashorn/src/jdk/nashorn/internal/ir/LiteralNode.java index ae80214c65a..5d41c9562d6 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/LiteralNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/LiteralNode.java @@ -659,9 +659,12 @@ public abstract class LiteralNode extends Node implements PropertyKey { * Copy constructor * @param node source array literal node */ - protected ArrayLiteralNode(final ArrayLiteralNode node) { - super(node); + private ArrayLiteralNode(final ArrayLiteralNode node, final Node[] value) { + super(node, value); this.elementType = node.elementType; + this.presets = node.presets; + this.postsets = node.postsets; + this.units = node.units; } /** @@ -750,9 +753,8 @@ public abstract class LiteralNode extends Node implements PropertyKey { break; } - final Symbol symbol = node.getSymbol(); - assert symbol != null; //don't run this on unresolved nodes or you are in trouble - Type symbolType = symbol.getSymbolType(); + assert node.getSymbol() != null; //don't run this on unresolved nodes or you are in trouble + Type symbolType = node.getSymbol().getSymbolType(); if (symbolType.isUnknown()) { symbolType = Type.OBJECT; } @@ -813,7 +815,8 @@ public abstract class LiteralNode extends Node implements PropertyKey { } /** - * Get indices of arrays containing computed post sets + * Get indices of arrays containing computed post sets. post sets + * are things like non literals e.g. "x+y" instead of i or 17 * @return post set indices */ public int[] getPostsets() { @@ -849,17 +852,17 @@ public abstract class LiteralNode extends Node implements PropertyKey { @Override public Node accept(final NodeVisitor visitor) { if (visitor.enterLiteralNode(this)) { - for (int i = 0; i < value.length; i++) { - final Node element = value[i]; - if (element != null) { - value[i] = element.accept(visitor); - } - } - return visitor.leaveLiteralNode(this); + final List oldValue = Arrays.asList(value); + final List newValue = Node.accept(visitor, Node.class, oldValue); + return visitor.leaveLiteralNode(oldValue != newValue ? setValue(newValue) : this); } return this; } + private ArrayLiteralNode setValue(final List value) { + return new ArrayLiteralNode(this, value.toArray(new Node[value.size()])); + } + @Override public void toString(final StringBuilder sb) { sb.append('['); diff --git a/nashorn/src/jdk/nashorn/internal/ir/Node.java b/nashorn/src/jdk/nashorn/internal/ir/Node.java index dfed903d956..d370f22f12f 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/Node.java +++ b/nashorn/src/jdk/nashorn/internal/ir/Node.java @@ -252,10 +252,17 @@ public abstract class Node extends Location { * Assign a symbol to this node. See {@link Node#getSymbol()} for explanation * of what a symbol is * + * @param lc lexical context * @param symbol the symbol + * @return new node */ - public void setSymbol(final Symbol symbol) { - this.symbol = symbol; + public Node setSymbol(final LexicalContext lc, final Symbol symbol) { + if (this.symbol == symbol) { + return this; + } + final Node newNode = (Node)clone(); + newNode.symbol = symbol; + return newNode; } /** @@ -274,7 +281,7 @@ public abstract class Node extends Location { final List newList = new ArrayList<>(); for (final Node node : list) { - final T newNode = clazz.cast(node.accept(visitor)); + final T newNode = node == null ? null : clazz.cast(node.accept(visitor)); if (newNode != node) { changed = true; } diff --git a/nashorn/src/jdk/nashorn/internal/ir/Symbol.java b/nashorn/src/jdk/nashorn/internal/ir/Symbol.java index da22f64d032..e776cd5cbb6 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/Symbol.java +++ b/nashorn/src/jdk/nashorn/internal/ir/Symbol.java @@ -67,6 +67,8 @@ public final class Symbol implements Comparable { public static final int IS_INTERNAL = 1 << 9; /** Is this a function self-reference symbol */ public static final int IS_FUNCTION_SELF = 1 << 10; + /** Is this a specialized param? */ + public static final int IS_SPECIALIZED_PARAM = 1 << 11; /** Null or name identifying symbol. */ private final String name; @@ -383,6 +385,15 @@ public final class Symbol implements Comparable { return (flags & KINDMASK) == IS_PARAM; } + /** + * Check if this symbol is a function parameter of known + * narrowest type + * @return true if parameter + */ + public boolean isSpecializedParam() { + return (flags & IS_SPECIALIZED_PARAM) == IS_SPECIALIZED_PARAM; + } + /** * Check whether this symbol ever has primitive assignments. Conservative * @return true if primitive assignments exist diff --git a/nashorn/src/jdk/nashorn/internal/objects/NativeRegExp.java b/nashorn/src/jdk/nashorn/internal/objects/NativeRegExp.java index 9a12911f155..8b469bc1ced 100644 --- a/nashorn/src/jdk/nashorn/internal/objects/NativeRegExp.java +++ b/nashorn/src/jdk/nashorn/internal/objects/NativeRegExp.java @@ -794,15 +794,15 @@ public final class NativeRegExp extends ScriptObject { RegExpResult match; final int inputLength = string.length(); - int lastLength = -1; - int lastIndex = 0; - int lastLastIndex = 0; + int splitLastLength = -1; + int splitLastIndex = 0; + int splitLastLastIndex = 0; - while ((match = execSplit(string, lastIndex)) != null) { - lastIndex = match.getIndex() + match.length(); + while ((match = execSplit(string, splitLastIndex)) != null) { + splitLastIndex = match.getIndex() + match.length(); - if (lastIndex > lastLastIndex) { - matches.add(string.substring(lastLastIndex, match.getIndex())); + if (splitLastIndex > splitLastLastIndex) { + matches.add(string.substring(splitLastLastIndex, match.getIndex())); final Object[] groups = match.getGroups(); if (groups.length > 1 && match.getIndex() < inputLength) { for (int index = 1; index < groups.length && matches.size() < limit; index++) { @@ -810,7 +810,7 @@ public final class NativeRegExp extends ScriptObject { } } - lastLength = match.length(); + splitLastLength = match.length(); if (matches.size() >= limit) { break; @@ -818,10 +818,10 @@ public final class NativeRegExp extends ScriptObject { } // bump the index to avoid infinite loop - if (lastIndex == lastLastIndex) { - lastIndex++; + if (splitLastIndex == splitLastLastIndex) { + splitLastIndex++; } else { - lastLastIndex = lastIndex; + splitLastLastIndex = splitLastIndex; } } @@ -829,12 +829,12 @@ public final class NativeRegExp extends ScriptObject { // check special case if we need to append an empty string at the // end of the match // if the lastIndex was the entire string - if (lastLastIndex == string.length()) { - if (lastLength > 0 || execSplit("", 0) == null) { + if (splitLastLastIndex == string.length()) { + if (splitLastLength > 0 || execSplit("", 0) == null) { matches.add(""); } } else { - matches.add(string.substring(lastLastIndex, inputLength)); + matches.add(string.substring(splitLastLastIndex, inputLength)); } } @@ -899,10 +899,6 @@ public final class NativeRegExp extends ScriptObject { } } - private void setGlobal(final boolean global) { - regexp.setGlobal(global); - } - boolean getGlobal() { return regexp.isGlobal(); } diff --git a/nashorn/src/jdk/nashorn/internal/parser/AbstractParser.java b/nashorn/src/jdk/nashorn/internal/parser/AbstractParser.java index 8f65e0f702e..b87d9cf3fea 100644 --- a/nashorn/src/jdk/nashorn/internal/parser/AbstractParser.java +++ b/nashorn/src/jdk/nashorn/internal/parser/AbstractParser.java @@ -249,6 +249,7 @@ public abstract class AbstractParser { * * @param errorType The error type of the warning * @param message Warning message. + * @param errorToken error token */ protected final void warning(final JSErrorType errorType, final String message, final long errorToken) { errors.warning(error(errorType, message, errorToken)); diff --git a/nashorn/src/jdk/nashorn/internal/parser/Parser.java b/nashorn/src/jdk/nashorn/internal/parser/Parser.java index ab70859b5ec..6302d1f7b03 100644 --- a/nashorn/src/jdk/nashorn/internal/parser/Parser.java +++ b/nashorn/src/jdk/nashorn/internal/parser/Parser.java @@ -305,6 +305,11 @@ loop: if (isStrictMode) { flags |= FunctionNode.IS_STRICT; } + if (env._specialize_calls != null) { + if (env._specialize_calls.contains(name)) { + flags |= FunctionNode.CAN_SPECIALIZE; + } + } // Start new block. FunctionNode functionNode = @@ -320,11 +325,11 @@ loop: kind, flags); - functionNode = functionNode.setState(lc, errors.hasErrors() ? CompilationState.PARSE_ERROR : CompilationState.PARSED); lc.push(functionNode); // Create new block, and just put it on the context stack, restoreFunctionNode() will associate it with the // FunctionNode. newBlock(); + return functionNode; } @@ -332,14 +337,19 @@ loop: * Restore the current block. */ private Block restoreBlock(final Block block) { - return lc.pop(block);//.setFlag(lc, flags); + return lc.pop(block); } + private FunctionNode restoreFunctionNode(final FunctionNode functionNode, final long lastToken) { final Block newBody = restoreBlock(lc.getFunctionBody(functionNode)); - return lc.pop(functionNode).setBody(lc, newBody).setLastToken(lc, lastToken); - } + return lc.pop(functionNode). + setBody(lc, newBody). + setLastToken(lc, lastToken). + setState(lc, errors.hasErrors() ? CompilationState.PARSE_ERROR : CompilationState.PARSED). + snapshot(lc); + } /** * Get the statements in a block. @@ -529,6 +539,7 @@ loop: script = restoreFunctionNode(script, token); //commit code script = script.setBody(lc, script.getBody().setNeedsScope(lc)); + return script; } @@ -800,7 +811,6 @@ loop: * @param ident Identifier that is verified * @param contextString String used in error message to give context to the user */ - @SuppressWarnings("fallthrough") private void verifyStrictIdent(final IdentNode ident, final String contextString) { if (isStrictMode) { switch (ident.getName()) { diff --git a/nashorn/src/jdk/nashorn/internal/runtime/CompiledFunction.java b/nashorn/src/jdk/nashorn/internal/runtime/CompiledFunction.java index 3cc9f09d238..d36f216c2b0 100644 --- a/nashorn/src/jdk/nashorn/internal/runtime/CompiledFunction.java +++ b/nashorn/src/jdk/nashorn/internal/runtime/CompiledFunction.java @@ -88,7 +88,7 @@ final class CompiledFunction implements Comparable { int weight = Type.typeFor(type.returnType()).getWeight(); for (final Class paramType : type.parameterArray()) { - final int pweight = Type.typeFor(paramType).getWeight(); + final int pweight = Type.typeFor(paramType).getWeight() * 2; //params are more important than call types as return values are always specialized weight += pweight; } return weight; diff --git a/nashorn/src/jdk/nashorn/internal/runtime/CompiledFunctions.java b/nashorn/src/jdk/nashorn/internal/runtime/CompiledFunctions.java index ff660d3d3b4..1ddc42b562b 100644 --- a/nashorn/src/jdk/nashorn/internal/runtime/CompiledFunctions.java +++ b/nashorn/src/jdk/nashorn/internal/runtime/CompiledFunctions.java @@ -69,5 +69,4 @@ final class CompiledFunctions extends TreeSet { return best(type).moreGenericThan(type); } - } diff --git a/nashorn/src/jdk/nashorn/internal/runtime/Context.java b/nashorn/src/jdk/nashorn/internal/runtime/Context.java index c3d32d0ba72..28b1a60a685 100644 --- a/nashorn/src/jdk/nashorn/internal/runtime/Context.java +++ b/nashorn/src/jdk/nashorn/internal/runtime/Context.java @@ -411,7 +411,7 @@ public final class Context { return ScriptRuntime.apply(func, evalThis); } - private Source loadInternal(final String srcStr, final String prefix, final String resourcePath) { + private static Source loadInternal(final String srcStr, final String prefix, final String resourcePath) { if (srcStr.startsWith(prefix)) { final String resource = resourcePath + srcStr.substring(prefix.length()); // NOTE: even sandbox scripts should be able to load scripts in nashorn: scheme @@ -759,10 +759,10 @@ public final class Context { final CodeSource cs = url == null ? null : new CodeSource(url, (CodeSigner[])null); final CodeInstaller installer = new ContextCodeInstaller(this, loader, cs); - final Compiler compiler = new Compiler(installer, functionNode, strict); + final Compiler compiler = new Compiler(installer, strict); - compiler.compile(); - script = compiler.install(); + final FunctionNode newFunctionNode = compiler.compile(functionNode); + script = compiler.install(newFunctionNode); if (global != null) { global.cacheClass(source, script); diff --git a/nashorn/src/jdk/nashorn/internal/runtime/JSONFunctions.java b/nashorn/src/jdk/nashorn/internal/runtime/JSONFunctions.java index 729df58cdb3..6106a314da2 100644 --- a/nashorn/src/jdk/nashorn/internal/runtime/JSONFunctions.java +++ b/nashorn/src/jdk/nashorn/internal/runtime/JSONFunctions.java @@ -25,8 +25,6 @@ package jdk.nashorn.internal.runtime; -import static jdk.nashorn.internal.runtime.ScriptObject.isArray; - import java.lang.invoke.MethodHandle; import java.util.Iterator; import java.util.List; diff --git a/nashorn/src/jdk/nashorn/internal/runtime/RecompilableScriptFunctionData.java b/nashorn/src/jdk/nashorn/internal/runtime/RecompilableScriptFunctionData.java index c2259896d13..1ba74f13db8 100644 --- a/nashorn/src/jdk/nashorn/internal/runtime/RecompilableScriptFunctionData.java +++ b/nashorn/src/jdk/nashorn/internal/runtime/RecompilableScriptFunctionData.java @@ -30,9 +30,12 @@ import static jdk.nashorn.internal.lookup.Lookup.MH; import java.lang.invoke.MethodHandle; import java.lang.invoke.MethodHandles; import java.lang.invoke.MethodType; +import java.util.LinkedList; + import jdk.nashorn.internal.codegen.Compiler; import jdk.nashorn.internal.codegen.CompilerConstants; import jdk.nashorn.internal.codegen.FunctionSignature; +import jdk.nashorn.internal.codegen.types.Type; import jdk.nashorn.internal.ir.FunctionNode; import jdk.nashorn.internal.ir.FunctionNode.CompilationState; import jdk.nashorn.internal.parser.Token; @@ -148,10 +151,10 @@ public final class RecompilableScriptFunctionData extends ScriptFunctionData { if (functionNode.isLazy()) { Compiler.LOG.info("Trampoline hit: need to do lazy compilation of '", functionNode.getName(), "'"); - final Compiler compiler = new Compiler(installer, functionNode); - functionNode = compiler.compile(); + final Compiler compiler = new Compiler(installer); + functionNode = compiler.compile(functionNode); assert !functionNode.isLazy(); - compiler.install(); + compiler.install(functionNode); // we don't need to update any flags - varArgs and needsCallee are instrincic // in the function world we need to get a destination node from the compile instead @@ -164,23 +167,114 @@ public final class RecompilableScriptFunctionData extends ScriptFunctionData { assert functionNode.hasState(CompilationState.EMITTED) : functionNode.getName() + " " + functionNode.getState() + " " + Debug.id(functionNode); // code exists - look it up and add it into the automatically sorted invoker list - code.add( - new CompiledFunction( - MH.findStatic( - LOOKUP, - functionNode.getCompileUnit().getCode(), - functionNode.getName(), - new FunctionSignature(functionNode). - getMethodType()))); + addCode(functionNode, null, null); } + private MethodHandle addCode(final FunctionNode fn, final MethodHandle guard, final MethodHandle fallback) { + final MethodHandle target = + MH.findStatic( + LOOKUP, + fn.getCompileUnit().getCode(), + fn.getName(), + new FunctionSignature(fn). + getMethodType()); + MethodHandle mh = target; + if (guard != null) { + try { + mh = MH.guardWithTest(MH.asCollector(guard, Object[].class, target.type().parameterCount()), MH.asType(target, fallback.type()), fallback); + } catch (Throwable e) { + e.printStackTrace(); + } + } + + final CompiledFunction cf = new CompiledFunction(mh); + code.add(cf); + + return cf.getInvoker(); + } + + private static Type runtimeType(final Object arg) { + if (arg == null) { + return Type.OBJECT; + } + + final Class clazz = arg.getClass(); + assert !clazz.isPrimitive() : "always boxed"; + if (clazz == Double.class) { + return JSType.isRepresentableAsInt((double)arg) ? Type.INT : Type.NUMBER; + } else if (clazz == Integer.class) { + return Type.INT; + } else if (clazz == Long.class) { + return Type.LONG; + } else if (clazz == String.class) { + return Type.STRING; + } + return Type.OBJECT; + } + + @SuppressWarnings("unused") + private static boolean paramTypeGuard(final Type[] compileTimeTypes, final Type[] runtimeTypes, Object... args) { + //System.err.println("Param type guard " + Arrays.asList(args)); + return false; + } + + private static final MethodHandle PARAM_TYPE_GUARD = findOwnMH("paramTypeGuard", boolean.class, Type[].class, Type[].class, Object[].class); + @Override MethodHandle getBestInvoker(final MethodType callSiteType, final Object[] args) { final MethodHandle mh = super.getBestInvoker(callSiteType, args); - if (code.isLessSpecificThan(callSiteType)) { - // opportunity for code specialization - we can regenerate a better version of this method + + if (!functionNode.canSpecialize() || !code.isLessSpecificThan(callSiteType)) { + return mh; } - return mh; + + final FunctionNode snapshot = functionNode.getSnapshot(); + int i; + + //classes known at runtime + final LinkedList runtimeArgs = new LinkedList<>(); + for (i = args.length - 1; i >= args.length - snapshot.getParameters().size(); i--) { + runtimeArgs.addLast(runtimeType(args[i])); + } + + //classes known at compile time + final LinkedList compileTimeArgs = new LinkedList<>(); + for (i = callSiteType.parameterCount() - 1; i >= 0 && compileTimeArgs.size() < snapshot.getParameters().size(); i--) { + compileTimeArgs.addLast(Type.typeFor(callSiteType.parameterType(i))); + } + + //the classes known at compile time are a safe to generate as primitives without parameter guards + //the classes known at runtime are safe to generate as primitives IFF there are parameter guards + MethodHandle guard = null; + for (i = 0; i < compileTimeArgs.size(); i++) { + final Type runtimeType = runtimeArgs.get(i); + final Type compileType = compileTimeArgs.get(i); + + if (compileType.isObject() && !runtimeType.isObject()) { + if (guard == null) { + guard = PARAM_TYPE_GUARD; + guard = MH.insertArguments(guard, 0, compileTimeArgs.toArray(new Type[compileTimeArgs.size()]), runtimeArgs.toArray(new Type[runtimeArgs.size()])); + } + } + } + + //System.err.println("Specialized " + name + " " + runtimeArgs + " known=" + compileTimeArgs); + + assert snapshot != null; + assert snapshot != functionNode; + + final Compiler compiler = new Compiler(installer); + final FunctionNode compiledSnapshot = compiler.compile(snapshot.setHints(null, new Compiler.Hints(compileTimeArgs.toArray(new Type[compileTimeArgs.size()])))); + + compiler.install(compiledSnapshot); + + final MethodHandle nmh = addCode(compiledSnapshot, guard, mh); + + return nmh; + } + + private static MethodHandle findOwnMH(final String name, final Class rtype, final Class... types) { + return MH.findStatic(MethodHandles.lookup(), RecompilableScriptFunctionData.class, name, MH.type(rtype, types)); } } diff --git a/nashorn/src/jdk/nashorn/internal/runtime/ScriptEnvironment.java b/nashorn/src/jdk/nashorn/internal/runtime/ScriptEnvironment.java index 74fab0cebe3..d03509e882c 100644 --- a/nashorn/src/jdk/nashorn/internal/runtime/ScriptEnvironment.java +++ b/nashorn/src/jdk/nashorn/internal/runtime/ScriptEnvironment.java @@ -26,9 +26,13 @@ package jdk.nashorn.internal.runtime; import java.io.PrintWriter; +import java.util.HashSet; import java.util.List; import java.util.Locale; +import java.util.Set; +import java.util.StringTokenizer; import java.util.TimeZone; + import jdk.nashorn.internal.codegen.Namespace; import jdk.nashorn.internal.runtime.linker.NashornCallSiteDescriptor; import jdk.nashorn.internal.runtime.options.KeyValueOption; @@ -151,6 +155,9 @@ public final class ScriptEnvironment { /** is this environment in scripting mode? */ public final boolean _scripting; + /** is the JIT allowed to specializ calls based on callsite types? */ + public final Set _specialize_calls; + /** is this environment in strict mode? */ public final boolean _strict; @@ -213,6 +220,17 @@ public final class ScriptEnvironment { _version = options.getBoolean("version"); _verify_code = options.getBoolean("verify.code"); + final String specialize = options.getString("specialize.calls"); + if (specialize == null) { + _specialize_calls = null; + } else { + _specialize_calls = new HashSet<>(); + final StringTokenizer st = new StringTokenizer(specialize, ","); + while (st.hasMoreElements()) { + _specialize_calls.add(st.nextToken()); + } + } + int callSiteFlags = 0; if (options.getBoolean("profile.callsites")) { callSiteFlags |= NashornCallSiteDescriptor.CALLSITE_PROFILE; @@ -246,6 +264,18 @@ public final class ScriptEnvironment { this._locale = Locale.getDefault(); } + /** + * Can we specialize a particular method name? + * @param functionName method name + * @return true if we are allowed to generate versions of this method + */ + public boolean canSpecialize(final String functionName) { + if (_specialize_calls == null) { + return false; + } + return _specialize_calls.isEmpty() || _specialize_calls.contains(functionName); + } + /** * Get the output stream for this environment * @return output print writer diff --git a/nashorn/src/jdk/nashorn/internal/runtime/ScriptObject.java b/nashorn/src/jdk/nashorn/internal/runtime/ScriptObject.java index 630bbbd0ca4..01d61d7f510 100644 --- a/nashorn/src/jdk/nashorn/internal/runtime/ScriptObject.java +++ b/nashorn/src/jdk/nashorn/internal/runtime/ScriptObject.java @@ -662,9 +662,9 @@ public abstract class ScriptObject extends PropertyListenerManager implements Pr } if (deep) { - final ScriptObject proto = getProto(); - if(proto != null) { - return proto.findProperty(key, deep, stopOnNonScope, start); + final ScriptObject myProto = getProto(); + if (myProto != null) { + return myProto.findProperty(key, deep, stopOnNonScope, start); } } diff --git a/nashorn/src/jdk/nashorn/internal/runtime/regexp/DefaultRegExp.java b/nashorn/src/jdk/nashorn/internal/runtime/regexp/DefaultRegExp.java index ecbc8bc9340..e3157ba5cb8 100644 --- a/nashorn/src/jdk/nashorn/internal/runtime/regexp/DefaultRegExp.java +++ b/nashorn/src/jdk/nashorn/internal/runtime/regexp/DefaultRegExp.java @@ -107,16 +107,16 @@ public class DefaultRegExp extends RegExp { class DefaultMatcher implements RegExpMatcher { final String input; - final Matcher matcher; + final Matcher defaultMatcher; DefaultMatcher(final String input) { this.input = input; - this.matcher = pattern.matcher(input); + this.defaultMatcher = pattern.matcher(input); } @Override public boolean search(final int start) { - return matcher.find(start); + return defaultMatcher.find(start); } @Override @@ -126,37 +126,37 @@ public class DefaultRegExp extends RegExp { @Override public int start() { - return matcher.start(); + return defaultMatcher.start(); } @Override public int start(final int group) { - return matcher.start(group); + return defaultMatcher.start(group); } @Override public int end() { - return matcher.end(); + return defaultMatcher.end(); } @Override public int end(final int group) { - return matcher.end(group); + return defaultMatcher.end(group); } @Override public String group() { - return matcher.group(); + return defaultMatcher.group(); } @Override public String group(final int group) { - return matcher.group(group); + return defaultMatcher.group(group); } @Override public int groupCount() { - return matcher.groupCount(); + return defaultMatcher.groupCount(); } } diff --git a/nashorn/src/jdk/nashorn/internal/runtime/regexp/JoniRegExp.java b/nashorn/src/jdk/nashorn/internal/runtime/regexp/JoniRegExp.java index 719f6c398ca..1047221af69 100644 --- a/nashorn/src/jdk/nashorn/internal/runtime/regexp/JoniRegExp.java +++ b/nashorn/src/jdk/nashorn/internal/runtime/regexp/JoniRegExp.java @@ -121,16 +121,16 @@ public class JoniRegExp extends RegExp { class JoniMatcher implements RegExpMatcher { final String input; - final Matcher matcher; + final Matcher joniMatcher; JoniMatcher(final String input) { this.input = input; - this.matcher = regex.matcher(input.toCharArray()); + this.joniMatcher = regex.matcher(input.toCharArray()); } @Override public boolean search(final int start) { - return matcher.search(start, input.length(), Option.NONE) > -1; + return joniMatcher.search(start, input.length(), Option.NONE) > -1; } @Override @@ -140,27 +140,27 @@ public class JoniRegExp extends RegExp { @Override public int start() { - return matcher.getBegin(); + return joniMatcher.getBegin(); } @Override public int start(final int group) { - return group == 0 ? start() : matcher.getRegion().beg[group]; + return group == 0 ? start() : joniMatcher.getRegion().beg[group]; } @Override public int end() { - return matcher.getEnd(); + return joniMatcher.getEnd(); } @Override public int end(final int group) { - return group == 0 ? end() : matcher.getRegion().end[group]; + return group == 0 ? end() : joniMatcher.getRegion().end[group]; } @Override public String group() { - return input.substring(matcher.getBegin(), matcher.getEnd()); + return input.substring(joniMatcher.getBegin(), joniMatcher.getEnd()); } @Override @@ -168,13 +168,13 @@ public class JoniRegExp extends RegExp { if (group == 0) { return group(); } - final Region region = matcher.getRegion(); + final Region region = joniMatcher.getRegion(); return input.substring(region.beg[group], region.end[group]); } @Override public int groupCount() { - final Region region = matcher.getRegion(); + final Region region = joniMatcher.getRegion(); return region == null ? 0 : region.numRegs - 1; } } diff --git a/nashorn/src/jdk/nashorn/internal/runtime/regexp/RegExpScanner.java b/nashorn/src/jdk/nashorn/internal/runtime/regexp/RegExpScanner.java index e651c44e126..31ee37b174e 100644 --- a/nashorn/src/jdk/nashorn/internal/runtime/regexp/RegExpScanner.java +++ b/nashorn/src/jdk/nashorn/internal/runtime/regexp/RegExpScanner.java @@ -934,7 +934,7 @@ final class RegExpScanner extends Scanner { return true; } - private void unicode(final int value, final StringBuilder buffer) { + private static void unicode(final int value, final StringBuilder buffer) { final String hex = Integer.toHexString(value); buffer.append('u'); for (int i = 0; i < 4 - hex.length(); i++) { @@ -944,7 +944,7 @@ final class RegExpScanner extends Scanner { } // Convert what would have been a backreference into a unicode escape, or a number literal, or both. - private void octalOrLiteral(final String numberLiteral, final StringBuilder buffer) { + private static void octalOrLiteral(final String numberLiteral, final StringBuilder buffer) { final int length = numberLiteral.length(); int octalValue = 0; int pos = 0; diff --git a/nashorn/src/jdk/nashorn/internal/runtime/resources/Options.properties b/nashorn/src/jdk/nashorn/internal/runtime/resources/Options.properties index 2b52e8b95b0..56bc412265c 100644 --- a/nashorn/src/jdk/nashorn/internal/runtime/resources/Options.properties +++ b/nashorn/src/jdk/nashorn/internal/runtime/resources/Options.properties @@ -288,12 +288,12 @@ nashorn.option.scripting = { \ dependency="--anon-functions=true" \ } -nashorn.option.timezone = { \ - name="-timezone", \ - short_name="-t", \ - params="", \ - desc="Set timezone for script execution.", \ - type=TimeZone \ +nashorn.option.specialize.calls = { \ + name="--specialize-calls", \ + is_undocumented=true, \ + type=String, \ + params="[=function_1,...,function_n]", \ + desc="Specialize all or a set of method according to callsite parameter types" \ } nashorn.option.stdout = { \ @@ -312,6 +312,14 @@ nashorn.option.stderr = { \ desc="Redirect stderr to a filename or to another tty, e.g. stdout" \ } +nashorn.option.timezone = { \ + name="-timezone", \ + short_name="-t", \ + params="", \ + desc="Set timezone for script execution.", \ + type=TimeZone \ +} + nashorn.option.trace.callsites = { \ name="--trace-callsites", \ short_name="-tcs", \ diff --git a/nashorn/src/jdk/nashorn/tools/Shell.java b/nashorn/src/jdk/nashorn/tools/Shell.java index 20725c0d3f2..8f5ee07e4f2 100644 --- a/nashorn/src/jdk/nashorn/tools/Shell.java +++ b/nashorn/src/jdk/nashorn/tools/Shell.java @@ -270,7 +270,7 @@ public class Shell { } //null - pass no code installer - this is compile only - new Compiler(env, functionNode).compile(); + new Compiler(env).compile(functionNode); } } finally { env.getOut().flush(); diff --git a/nashorn/test/script/basic/paramspec.js b/nashorn/test/script/basic/paramspec.js new file mode 100644 index 00000000000..c0ea2a549fa --- /dev/null +++ b/nashorn/test/script/basic/paramspec.js @@ -0,0 +1,38 @@ +/* + * Copyright (c) 2010, 2013, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/** + * paramspec - test that Attr doesn't break parameters when specializing + * + * @run + * @test + */ + +function f(a) { + var b = ~a; + return b == ~17; +} + +print(f("17")); +print(f(17)); + diff --git a/nashorn/test/script/basic/paramspec.js.EXPECTED b/nashorn/test/script/basic/paramspec.js.EXPECTED new file mode 100644 index 00000000000..bb101b641b9 --- /dev/null +++ b/nashorn/test/script/basic/paramspec.js.EXPECTED @@ -0,0 +1,2 @@ +true +true diff --git a/nashorn/test/script/basic/runsunspider.js b/nashorn/test/script/basic/runsunspider.js index 7f787975888..149bd27bc81 100644 --- a/nashorn/test/script/basic/runsunspider.js +++ b/nashorn/test/script/basic/runsunspider.js @@ -86,7 +86,7 @@ function runsuite(tests) { changed = true; } } catch (e) { - print("error: " + e); + print("error: " + e.printStackTrace()); if (e.toString().indexOf(tests) == 1) { throw e; } diff --git a/nashorn/test/script/trusted/logcoverage.js b/nashorn/test/script/currently-failing/logcoverage.js similarity index 94% rename from nashorn/test/script/trusted/logcoverage.js rename to nashorn/test/script/currently-failing/logcoverage.js index a2cd7a0f103..18b16aa1e00 100644 --- a/nashorn/test/script/trusted/logcoverage.js +++ b/nashorn/test/script/currently-failing/logcoverage.js @@ -90,16 +90,17 @@ var methodsCalled = [ function check(str, strs) { for each (s in strs) { - if (s.indexOf(str) !== -1) { + if (str.indexOf(s) !== -1) { continue; } - print(method + "not found"); + print(s + " not found"); return; } print("check ok!"); } str = runScriptEngine([ "--log=codegen,compiler=finest,methodhandles=finest,fields=finest" ], __DIR__ + "../basic/NASHORN-19.js"); +str += runScriptEngine([ "--log=codegen,compiler=finest,methodhandles=finest,fields=finest" ], __DIR__ + "../basic/varargs.js"); check(str, methodsCalled); check(str, ['return', 'get', 'set', '[fields]']); From 87ff1bd75194f0cecd450cd6678f5f75c5eecb64 Mon Sep 17 00:00:00 2001 From: Marcus Lagergren Date: Fri, 3 May 2013 16:01:33 +0200 Subject: [PATCH 06/15] 8013871: mem usage histograms enabled with compiler logging level set to more specific than or equals to info when --print-mem-usage flag is used Reviewed-by: jlaskey, hannesw --- .../nashorn/internal/codegen/Compiler.java | 43 ++ .../ir/debug/ClassHistogramElement.java | 113 +++++ .../ir/debug/ObjectSizeCalculator.java | 457 ++++++++++++++++++ .../jdk/nashorn/internal/objects/Global.java | 1 - .../jdk/nashorn/internal/runtime/Context.java | 1 - .../internal/runtime/ScriptEnvironment.java | 4 + .../internal/runtime/options/Options.java | 2 +- .../runtime/resources/Options.properties | 6 + nashorn/src/jdk/nashorn/tools/Shell.java | 4 +- 9 files changed, 626 insertions(+), 5 deletions(-) create mode 100644 nashorn/src/jdk/nashorn/internal/ir/debug/ClassHistogramElement.java create mode 100644 nashorn/src/jdk/nashorn/internal/ir/debug/ObjectSizeCalculator.java diff --git a/nashorn/src/jdk/nashorn/internal/codegen/Compiler.java b/nashorn/src/jdk/nashorn/internal/codegen/Compiler.java index 2f0c0d2272b..247e560ed1e 100644 --- a/nashorn/src/jdk/nashorn/internal/codegen/Compiler.java +++ b/nashorn/src/jdk/nashorn/internal/codegen/Compiler.java @@ -42,10 +42,13 @@ import java.security.AccessController; import java.security.PrivilegedActionException; import java.security.PrivilegedExceptionAction; import java.util.Arrays; +import java.util.Collections; +import java.util.Comparator; import java.util.EnumSet; import java.util.HashMap; import java.util.HashSet; import java.util.LinkedList; +import java.util.List; import java.util.Map; import java.util.Map.Entry; import java.util.Set; @@ -56,6 +59,8 @@ import jdk.nashorn.internal.codegen.ClassEmitter.Flag; import jdk.nashorn.internal.codegen.types.Type; import jdk.nashorn.internal.ir.FunctionNode; import jdk.nashorn.internal.ir.FunctionNode.CompilationState; +import jdk.nashorn.internal.ir.debug.ClassHistogramElement; +import jdk.nashorn.internal.ir.debug.ObjectSizeCalculator; import jdk.nashorn.internal.runtime.CodeInstaller; import jdk.nashorn.internal.runtime.DebugLogger; import jdk.nashorn.internal.runtime.ScriptEnvironment; @@ -289,6 +294,40 @@ public final class Compiler { this(env, null, sequence(env._lazy_compilation), env._strict); } + private static void printMemoryUsage(final String phaseName, final FunctionNode functionNode) { + final ObjectSizeCalculator osc = new ObjectSizeCalculator(ObjectSizeCalculator.getEffectiveMemoryLayoutSpecification()); + osc.calculateObjectSize(functionNode); + + final List list = osc.getClassHistogram(); + + final StringBuilder sb = new StringBuilder(); + final long totalSize = osc.calculateObjectSize(functionNode); + sb.append(phaseName).append(" Total size = ").append(totalSize / 1024 / 1024).append("MB"); + LOG.info(sb); + + Collections.sort(list, new Comparator() { + @Override + public int compare(ClassHistogramElement o1, ClassHistogramElement o2) { + final long diff = o1.getBytes() - o2.getBytes(); + if (diff < 0) { + return 1; + } else if (diff > 0) { + return -1; + } else { + return 0; + } + } + }); + for (final ClassHistogramElement e : list) { + final String line = String.format(" %-48s %10d bytes (%8d instances)", e.getClazz(), e.getBytes(), e.getInstances()); + LOG.info(line); + if (e.getBytes() < totalSize / 20) { + LOG.info(" ..."); + break; // never mind, so little memory anyway + } + } + } + /** * Execute the compilation this Compiler was created with * @param functionNode function node to compile from its current state @@ -312,6 +351,10 @@ public final class Compiler { for (final CompilationPhase phase : sequence) { newFunctionNode = phase.apply(this, newFunctionNode); + if (env._print_mem_usage) { + printMemoryUsage(phase.toString(), newFunctionNode); + } + final long duration = Timing.isEnabled() ? (phase.getEndTime() - phase.getStartTime()) : 0L; time += duration; diff --git a/nashorn/src/jdk/nashorn/internal/ir/debug/ClassHistogramElement.java b/nashorn/src/jdk/nashorn/internal/ir/debug/ClassHistogramElement.java new file mode 100644 index 00000000000..c65f8ef7ce7 --- /dev/null +++ b/nashorn/src/jdk/nashorn/internal/ir/debug/ClassHistogramElement.java @@ -0,0 +1,113 @@ +/* + * Copyright (c) 2010, 2013, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +package jdk.nashorn.internal.ir.debug; + +import java.util.Comparator; + +/** + * Class histogram element for IR / Java object instrumentation + */ +public class ClassHistogramElement { + /** + * Instance comparator + */ + public static final Comparator COMPARE_INSTANCES = new Comparator() { + @Override + public int compare(final ClassHistogramElement o1, final ClassHistogramElement o2) { + return (int)Math.abs(o1.instances - o2.instances); + } + }; + + /** + * Bytes comparator + */ + public static final Comparator COMPARE_BYTES = new Comparator() { + @Override + public int compare(final ClassHistogramElement o1, final ClassHistogramElement o2) { + return (int)Math.abs(o1.bytes - o2.bytes); + } + }; + + /** + * Classname comparator + */ + public static final Comparator COMPARE_CLASSNAMES = new Comparator() { + @Override + public int compare(final ClassHistogramElement o1, final ClassHistogramElement o2) { + return o1.clazz.getCanonicalName().compareTo(o2.clazz.getCanonicalName()); + } + }; + + private final Class clazz; + private long instances; + private long bytes; + + /** + * Constructor + * @param clazz class for which to construct histogram + */ + public ClassHistogramElement(final Class clazz) { + this.clazz = clazz; + } + + /** + * Add an instance + * @param sizeInBytes byte count + */ + public void addInstance(final long sizeInBytes) { + instances++; + this.bytes += sizeInBytes; + } + + /** + * Get size in bytes + * @return size in bytes + */ + public long getBytes() { + return bytes; + } + + /** + * Get class + * @return class + */ + public Class getClazz() { + return clazz; + } + + /** + * Get number of instances + * @return number of instances + */ + public long getInstances() { + return instances; + } + + @Override + public String toString() { + return "ClassHistogramElement[class=" + clazz.getCanonicalName() + ", instances=" + instances + ", bytes=" + bytes + "]"; + } +} diff --git a/nashorn/src/jdk/nashorn/internal/ir/debug/ObjectSizeCalculator.java b/nashorn/src/jdk/nashorn/internal/ir/debug/ObjectSizeCalculator.java new file mode 100644 index 00000000000..2d4412e77ac --- /dev/null +++ b/nashorn/src/jdk/nashorn/internal/ir/debug/ObjectSizeCalculator.java @@ -0,0 +1,457 @@ +/* + * Copyright (c) 2010, 2013, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +package jdk.nashorn.internal.ir.debug; + +import java.lang.management.ManagementFactory; +import java.lang.management.MemoryPoolMXBean; +import java.lang.reflect.Array; +import java.lang.reflect.Field; +import java.lang.reflect.Modifier; +import java.util.ArrayDeque; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.Deque; +import java.util.IdentityHashMap; +import java.util.LinkedList; +import java.util.List; +import java.util.Map; + +/** + * Contains utility methods for calculating the memory usage of objects. It + * only works on the HotSpot JVM, and infers the actual memory layout (32 bit + * vs. 64 bit word size, compressed object pointers vs. uncompressed) from + * best available indicators. It can reliably detect a 32 bit vs. 64 bit JVM. + * It can only make an educated guess at whether compressed OOPs are used, + * though; specifically, it knows what the JVM's default choice of OOP + * compression would be based on HotSpot version and maximum heap sizes, but if + * the choice is explicitly overridden with the -XX:{+|-}UseCompressedOops command line + * switch, it can not detect + * this fact and will report incorrect sizes, as it will presume the default JVM + * behavior. + * + * @author Attila Szegedi + */ +public class ObjectSizeCalculator { + + /** + * Describes constant memory overheads for various constructs in a JVM implementation. + */ + public interface MemoryLayoutSpecification { + + /** + * Returns the fixed overhead of an array of any type or length in this JVM. + * + * @return the fixed overhead of an array. + */ + int getArrayHeaderSize(); + + /** + * Returns the fixed overhead of for any {@link Object} subclass in this JVM. + * + * @return the fixed overhead of any object. + */ + int getObjectHeaderSize(); + + /** + * Returns the quantum field size for a field owned by an object in this JVM. + * + * @return the quantum field size for an object. + */ + int getObjectPadding(); + + /** + * Returns the fixed size of an object reference in this JVM. + * + * @return the size of all object references. + */ + int getReferenceSize(); + + /** + * Returns the quantum field size for a field owned by one of an object's ancestor superclasses + * in this JVM. + * + * @return the quantum field size for a superclass field. + */ + int getSuperclassFieldPadding(); + } + + private static class CurrentLayout { + private static final MemoryLayoutSpecification SPEC = + getEffectiveMemoryLayoutSpecification(); + } + + /** + * Given an object, returns the total allocated size, in bytes, of the object + * and all other objects reachable from it. Attempts to to detect the current JVM memory layout, + * but may fail with {@link UnsupportedOperationException}; + * + * @param obj the object; can be null. Passing in a {@link java.lang.Class} object doesn't do + * anything special, it measures the size of all objects + * reachable through it (which will include its class loader, and by + * extension, all other Class objects loaded by + * the same loader, and all the parent class loaders). It doesn't provide the + * size of the static fields in the JVM class that the Class object + * represents. + * @return the total allocated size of the object and all other objects it + * retains. + * @throws UnsupportedOperationException if the current vm memory layout cannot be detected. + */ + public static long getObjectSize(final Object obj) throws UnsupportedOperationException { + return obj == null ? 0 : new ObjectSizeCalculator(CurrentLayout.SPEC).calculateObjectSize(obj); + } + + // Fixed object header size for arrays. + private final int arrayHeaderSize; + // Fixed object header size for non-array objects. + private final int objectHeaderSize; + // Padding for the object size - if the object size is not an exact multiple + // of this, it is padded to the next multiple. + private final int objectPadding; + // Size of reference (pointer) fields. + private final int referenceSize; + // Padding for the fields of superclass before fields of subclasses are + // added. + private final int superclassFieldPadding; + + private final Map, ClassSizeInfo> classSizeInfos = new IdentityHashMap<>(); + + + private final Map alreadyVisited = new IdentityHashMap<>(); + private final Map, ClassHistogramElement> histogram = new IdentityHashMap<>(); + + private final Deque pending = new ArrayDeque<>(16 * 1024); + private long size; + + /** + * Creates an object size calculator that can calculate object sizes for a given + * {@code memoryLayoutSpecification}. + * + * @param memoryLayoutSpecification a description of the JVM memory layout. + */ + public ObjectSizeCalculator(final MemoryLayoutSpecification memoryLayoutSpecification) { + memoryLayoutSpecification.getClass(); + arrayHeaderSize = memoryLayoutSpecification.getArrayHeaderSize(); + objectHeaderSize = memoryLayoutSpecification.getObjectHeaderSize(); + objectPadding = memoryLayoutSpecification.getObjectPadding(); + referenceSize = memoryLayoutSpecification.getReferenceSize(); + superclassFieldPadding = memoryLayoutSpecification.getSuperclassFieldPadding(); + } + + /** + * Given an object, returns the total allocated size, in bytes, of the object + * and all other objects reachable from it. + * + * @param obj the object; can be null. Passing in a {@link java.lang.Class} object doesn't do + * anything special, it measures the size of all objects + * reachable through it (which will include its class loader, and by + * extension, all other Class objects loaded by + * the same loader, and all the parent class loaders). It doesn't provide the + * size of the static fields in the JVM class that the Class object + * represents. + * @return the total allocated size of the object and all other objects it + * retains. + */ + public synchronized long calculateObjectSize(final Object obj) { + // Breadth-first traversal instead of naive depth-first with recursive + // implementation, so we don't blow the stack traversing long linked lists. + histogram.clear(); + try { + for (Object o = obj;;) { + visit(o); + if (pending.isEmpty()) { + return size; + } + o = pending.removeFirst(); + } + } finally { + alreadyVisited.clear(); + pending.clear(); + size = 0; + } + } + + /** + * Get the class histograpm + * @return class histogram element list + */ + public List getClassHistogram() { + return new ArrayList<>(histogram.values()); + } + + private ClassSizeInfo getClassSizeInfo(final Class clazz) { + ClassSizeInfo csi = classSizeInfos.get(clazz); + if(csi == null) { + csi = new ClassSizeInfo(clazz); + classSizeInfos.put(clazz, csi); + } + return csi; + } + + private void visit(final Object obj) { + if (alreadyVisited.containsKey(obj)) { + return; + } + final Class clazz = obj.getClass(); + if (clazz == ArrayElementsVisitor.class) { + ((ArrayElementsVisitor) obj).visit(this); + } else { + alreadyVisited.put(obj, obj); + if (clazz.isArray()) { + visitArray(obj); + } else { + getClassSizeInfo(clazz).visit(obj, this); + } + } + } + + private void visitArray(final Object array) { + final Class arrayClass = array.getClass(); + final Class componentType = arrayClass.getComponentType(); + final int length = Array.getLength(array); + if (componentType.isPrimitive()) { + increaseByArraySize(arrayClass, length, getPrimitiveFieldSize(componentType)); + } else { + increaseByArraySize(arrayClass, length, referenceSize); + // If we didn't use an ArrayElementsVisitor, we would be enqueueing every + // element of the array here instead. For large arrays, it would + // tremendously enlarge the queue. In essence, we're compressing it into + // a small command object instead. This is different than immediately + // visiting the elements, as their visiting is scheduled for the end of + // the current queue. + switch (length) { + case 0: { + break; + } + case 1: { + enqueue(Array.get(array, 0)); + break; + } + default: { + enqueue(new ArrayElementsVisitor((Object[]) array)); + } + } + } + } + + private void increaseByArraySize(final Class clazz, final int length, final long elementSize) { + increaseSize(clazz, roundTo(arrayHeaderSize + length * elementSize, objectPadding)); + } + + private static class ArrayElementsVisitor { + private final Object[] array; + + ArrayElementsVisitor(final Object[] array) { + this.array = array; + } + + public void visit(final ObjectSizeCalculator calc) { + for (final Object elem : array) { + if (elem != null) { + calc.visit(elem); + } + } + } + } + + void enqueue(final Object obj) { + if (obj != null) { + pending.addLast(obj); + } + } + + void increaseSize(final Class clazz, final long objectSize) { + ClassHistogramElement he = histogram.get(clazz); + if(he == null) { + he = new ClassHistogramElement(clazz); + histogram.put(clazz, he); + } + he.addInstance(objectSize); + size += objectSize; + } + + static long roundTo(final long x, final int multiple) { + return ((x + multiple - 1) / multiple) * multiple; + } + + private class ClassSizeInfo { + // Padded fields + header size + private final long objectSize; + // Only the fields size - used to calculate the subclasses' memory + // footprint. + private final long fieldsSize; + private final Field[] referenceFields; + + public ClassSizeInfo(final Class clazz) { + long newFieldsSize = 0; + final List newReferenceFields = new LinkedList<>(); + for (Field f : clazz.getDeclaredFields()) { + if (Modifier.isStatic(f.getModifiers())) { + continue; + } + final Class type = f.getType(); + if (type.isPrimitive()) { + newFieldsSize += getPrimitiveFieldSize(type); + } else { + f.setAccessible(true); + newReferenceFields.add(f); + newFieldsSize += referenceSize; + } + } + final Class superClass = clazz.getSuperclass(); + if (superClass != null) { + final ClassSizeInfo superClassInfo = getClassSizeInfo(superClass); + newFieldsSize += roundTo(superClassInfo.fieldsSize, superclassFieldPadding); + newReferenceFields.addAll(Arrays.asList(superClassInfo.referenceFields)); + } + this.fieldsSize = newFieldsSize; + this.objectSize = roundTo(objectHeaderSize + newFieldsSize, objectPadding); + this.referenceFields = newReferenceFields.toArray( + new Field[newReferenceFields.size()]); + } + + void visit(final Object obj, final ObjectSizeCalculator calc) { + calc.increaseSize(obj.getClass(), objectSize); + enqueueReferencedObjects(obj, calc); + } + + public void enqueueReferencedObjects(final Object obj, final ObjectSizeCalculator calc) { + for (Field f : referenceFields) { + try { + calc.enqueue(f.get(obj)); + } catch (IllegalAccessException e) { + final AssertionError ae = new AssertionError( + "Unexpected denial of access to " + f); + ae.initCause(e); + throw ae; + } + } + } + } + + private static long getPrimitiveFieldSize(final Class type) { + if (type == boolean.class || type == byte.class) { + return 1; + } + if (type == char.class || type == short.class) { + return 2; + } + if (type == int.class || type == float.class) { + return 4; + } + if (type == long.class || type == double.class) { + return 8; + } + throw new AssertionError("Encountered unexpected primitive type " + + type.getName()); + } + + /** + * Return the current memory usage + * @return current memory usage derived from system configuration + */ + public static MemoryLayoutSpecification getEffectiveMemoryLayoutSpecification() { + final String vmName = System.getProperty("java.vm.name"); + if (vmName == null || !vmName.startsWith("Java HotSpot(TM) ")) { + throw new UnsupportedOperationException( + "ObjectSizeCalculator only supported on HotSpot VM"); + } + + final String dataModel = System.getProperty("sun.arch.data.model"); + if ("32".equals(dataModel)) { + // Running with 32-bit data model + return new MemoryLayoutSpecification() { + @Override public int getArrayHeaderSize() { + return 12; + } + @Override public int getObjectHeaderSize() { + return 8; + } + @Override public int getObjectPadding() { + return 8; + } + @Override public int getReferenceSize() { + return 4; + } + @Override public int getSuperclassFieldPadding() { + return 4; + } + }; + } else if (!"64".equals(dataModel)) { + throw new UnsupportedOperationException("Unrecognized value '" + + dataModel + "' of sun.arch.data.model system property"); + } + + final String strVmVersion = System.getProperty("java.vm.version"); + final int vmVersion = Integer.parseInt(strVmVersion.substring(0, + strVmVersion.indexOf('.'))); + if (vmVersion >= 17) { + long maxMemory = 0; + for (MemoryPoolMXBean mp : ManagementFactory.getMemoryPoolMXBeans()) { + maxMemory += mp.getUsage().getMax(); + } + if (maxMemory < 30L * 1024 * 1024 * 1024) { + // HotSpot 17.0 and above use compressed OOPs below 30GB of RAM total + // for all memory pools (yes, including code cache). + return new MemoryLayoutSpecification() { + @Override public int getArrayHeaderSize() { + return 16; + } + @Override public int getObjectHeaderSize() { + return 12; + } + @Override public int getObjectPadding() { + return 8; + } + @Override public int getReferenceSize() { + return 4; + } + @Override public int getSuperclassFieldPadding() { + return 4; + } + }; + } + } + + // In other cases, it's a 64-bit uncompressed OOPs object model + return new MemoryLayoutSpecification() { + @Override public int getArrayHeaderSize() { + return 24; + } + @Override public int getObjectHeaderSize() { + return 16; + } + @Override public int getObjectPadding() { + return 8; + } + @Override public int getReferenceSize() { + return 8; + } + @Override public int getSuperclassFieldPadding() { + return 8; + } + }; + } +} diff --git a/nashorn/src/jdk/nashorn/internal/objects/Global.java b/nashorn/src/jdk/nashorn/internal/objects/Global.java index e079248b577..d139b5cd56e 100644 --- a/nashorn/src/jdk/nashorn/internal/objects/Global.java +++ b/nashorn/src/jdk/nashorn/internal/objects/Global.java @@ -1625,7 +1625,6 @@ public final class Global extends ScriptObject implements GlobalObject, Scope { this.addOwnProperty("Debug", Attribute.NOT_ENUMERABLE, initConstructor("Debug")); } - @SuppressWarnings("resource") private static Object printImpl(final boolean newLine, final Object... objects) { final PrintWriter out = Global.getEnv().getOut(); diff --git a/nashorn/src/jdk/nashorn/internal/runtime/Context.java b/nashorn/src/jdk/nashorn/internal/runtime/Context.java index 28b1a60a685..84113a748c5 100644 --- a/nashorn/src/jdk/nashorn/internal/runtime/Context.java +++ b/nashorn/src/jdk/nashorn/internal/runtime/Context.java @@ -171,7 +171,6 @@ public final class Context { * @param str text to write * @param crlf write a carriage return/new line after text */ - @SuppressWarnings("resource") public static void err(final String str, final boolean crlf) { final PrintWriter err = Context.getCurrentErr(); if (err != null) { diff --git a/nashorn/src/jdk/nashorn/internal/runtime/ScriptEnvironment.java b/nashorn/src/jdk/nashorn/internal/runtime/ScriptEnvironment.java index d03509e882c..0a556f9d51c 100644 --- a/nashorn/src/jdk/nashorn/internal/runtime/ScriptEnvironment.java +++ b/nashorn/src/jdk/nashorn/internal/runtime/ScriptEnvironment.java @@ -140,6 +140,9 @@ public final class ScriptEnvironment { /** Print resulting bytecode for script */ public final boolean _print_code; + /** Print memory usage for IR after each phase */ + public final boolean _print_mem_usage; + /** Print function will no print newline characters */ public final boolean _print_no_newline; @@ -211,6 +214,7 @@ public final class ScriptEnvironment { _print_ast = options.getBoolean("print.ast"); _print_lower_ast = options.getBoolean("print.lower.ast"); _print_code = options.getBoolean("print.code"); + _print_mem_usage = options.getBoolean("print.mem.usage"); _print_no_newline = options.getBoolean("print.no.newline"); _print_parse = options.getBoolean("print.parse"); _print_lower_parse = options.getBoolean("print.lower.parse"); diff --git a/nashorn/src/jdk/nashorn/internal/runtime/options/Options.java b/nashorn/src/jdk/nashorn/internal/runtime/options/Options.java index 16f3bc0f5ef..286feec907b 100644 --- a/nashorn/src/jdk/nashorn/internal/runtime/options/Options.java +++ b/nashorn/src/jdk/nashorn/internal/runtime/options/Options.java @@ -243,7 +243,7 @@ public final class Options { */ public String getString(final String key) { final Option option = get(key); - if(option != null) { + if (option != null) { final String value = (String)option.getValue(); if(value != null) { return value.intern(); diff --git a/nashorn/src/jdk/nashorn/internal/runtime/resources/Options.properties b/nashorn/src/jdk/nashorn/internal/runtime/resources/Options.properties index 56bc412265c..4717319710f 100644 --- a/nashorn/src/jdk/nashorn/internal/runtime/resources/Options.properties +++ b/nashorn/src/jdk/nashorn/internal/runtime/resources/Options.properties @@ -247,6 +247,12 @@ nashorn.option.print.code = { \ desc="Print bytecode." \ } +nashorn.option.print.mem.usage = { \ + name="--print-mem-usage", \ + is_undocumented=true, \ + desc="Print memory usage of IR after each compile stage." \ +} + nashorn.option.print.no.newline = { \ name="--print-no-newline", \ is_undocumented=true, \ diff --git a/nashorn/src/jdk/nashorn/tools/Shell.java b/nashorn/src/jdk/nashorn/tools/Shell.java index 8f5ee07e4f2..708fecc59f1 100644 --- a/nashorn/src/jdk/nashorn/tools/Shell.java +++ b/nashorn/src/jdk/nashorn/tools/Shell.java @@ -173,9 +173,9 @@ public class Shell { if (env._fx) { return runFXScripts(context, global, files); - } else { - return runScripts(context, global, files); } + + return runScripts(context, global, files); } /** From 973d65632061f620eeb44d0a83fff170fcb0ecd6 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Hannes=20Walln=C3=B6fer?= Date: Fri, 3 May 2013 22:47:23 +0200 Subject: [PATCH 07/15] 8013878: ClassCastException in Regex Reviewed-by: jlaskey --- .../nashorn/internal/objects/NativeArray.java | 19 +++---- nashorn/test/script/basic/JDK-8013878.js | 53 +++++++++++++++++++ .../test/script/basic/JDK-8013878.js.EXPECTED | 24 +++++++++ 3 files changed, 85 insertions(+), 11 deletions(-) create mode 100644 nashorn/test/script/basic/JDK-8013878.js create mode 100644 nashorn/test/script/basic/JDK-8013878.js.EXPECTED diff --git a/nashorn/src/jdk/nashorn/internal/objects/NativeArray.java b/nashorn/src/jdk/nashorn/internal/objects/NativeArray.java index a659af4feb9..c9a71b57a92 100644 --- a/nashorn/src/jdk/nashorn/internal/objects/NativeArray.java +++ b/nashorn/src/jdk/nashorn/internal/objects/NativeArray.java @@ -297,7 +297,7 @@ public final class NativeArray extends ScriptObject { @Getter(attributes = Attribute.NOT_ENUMERABLE | Attribute.NOT_CONFIGURABLE) public static Object length(final Object self) { if (isArray(self)) { - return ((NativeArray) self).getArray().length() & JSType.MAX_UINT; + return ((ScriptObject) self).getArray().length() & JSType.MAX_UINT; } return 0; @@ -311,7 +311,7 @@ public final class NativeArray extends ScriptObject { @Setter(attributes = Attribute.NOT_ENUMERABLE | Attribute.NOT_CONFIGURABLE) public static void length(final Object self, final Object length) { if (isArray(self)) { - ((NativeArray) self).setLength(validLength(length, true)); + ((ScriptObject) self).setLength(validLength(length, true)); } } @@ -642,10 +642,9 @@ public final class NativeArray extends ScriptObject { final boolean strict = sobj.isStrictContext(); if (bulkable(sobj)) { - final NativeArray nativeArray = (NativeArray)sobj; - if (nativeArray.getArray().length() + args.length <= JSType.MAX_UINT) { - final ArrayData newData = nativeArray.getArray().push(nativeArray.isStrictContext(), args); - nativeArray.setArray(newData); + if (sobj.getArray().length() + args.length <= JSType.MAX_UINT) { + final ArrayData newData = sobj.getArray().push(sobj.isStrictContext(), args); + sobj.setArray(newData); return newData.length(); } //fallthru @@ -780,8 +779,7 @@ public final class NativeArray extends ScriptObject { } if (bulkable(sobj)) { - final NativeArray narray = (NativeArray) sobj; - return new NativeArray(narray.getArray().slice(k, finale)); + return new NativeArray(sobj.getArray().slice(k, finale)); } final NativeArray copy = new NativeArray(0); @@ -1001,11 +999,10 @@ public final class NativeArray extends ScriptObject { } if (bulkable(sobj)) { - final NativeArray nativeArray = (NativeArray) sobj; - nativeArray.getArray().shiftRight(items.length); + sobj.getArray().shiftRight(items.length); for (int j = 0; j < items.length; j++) { - nativeArray.setArray(nativeArray.getArray().set(j, items[j], sobj.isStrictContext())); + sobj.setArray(sobj.getArray().set(j, items[j], sobj.isStrictContext())); } } else { for (long k = len; k > 0; k--) { diff --git a/nashorn/test/script/basic/JDK-8013878.js b/nashorn/test/script/basic/JDK-8013878.js new file mode 100644 index 00000000000..81e39f91bb0 --- /dev/null +++ b/nashorn/test/script/basic/JDK-8013878.js @@ -0,0 +1,53 @@ +/* + * Copyright (c) 2010, 2013, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/** + * JDK-8013878: ClassCastException in Regex + * + * @test + * @run + */ + +var re = /(a)(b)(c)/; +var str = 'abc'; + +print(re.exec(str).length); +print(re.exec(str).concat(['d', 'e', 'f'])); +print(re.exec(str).join('-')); +print(re.exec(str).push('d')); +print(re.exec(str).pop()); +print(re.exec(str).reverse()); +print(re.exec(str).shift()); +print(re.exec(str).sort()); +print(re.exec(str).slice(1)); +print(re.exec(str).splice(1, 2, 'foo')); +print(re.exec(str).unshift('x')); +print(re.exec(str).indexOf('a')); +print(re.exec(str).lastIndexOf('a')); +print(re.exec(str).every(function(a) {return a.length;})); +print(re.exec(str).some(function(a) {return a.length;})); +print(re.exec(str).filter(function(a) {return a.length;})); +print(re.exec(str).forEach(function(a) {print(a)})); +print(re.exec(str).map(function(a) {return a.length;})); +print(re.exec(str).reduce(function(a, b) {return a + b})); +print(re.exec(str).reduceRight(function(a, b) {return a + b})); diff --git a/nashorn/test/script/basic/JDK-8013878.js.EXPECTED b/nashorn/test/script/basic/JDK-8013878.js.EXPECTED new file mode 100644 index 00000000000..7dbcf439559 --- /dev/null +++ b/nashorn/test/script/basic/JDK-8013878.js.EXPECTED @@ -0,0 +1,24 @@ +4 +abc,a,b,c,d,e,f +abc-a-b-c +5 +c +c,b,a,abc +abc +a,abc,b,c +a,b,c +a,b +5 +1 +1 +true +true +abc,a,b,c +abc +a +b +c +undefined +3,1,1,1 +abcabc +cbaabc From 253e421ec582cef92ff3c6c98b7125fac35d83e3 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Hannes=20Walln=C3=B6fer?= Date: Fri, 3 May 2013 22:48:53 +0200 Subject: [PATCH 08/15] 8013873: Regexp regression for escaped dash in character class Reviewed-by: jlaskey --- .../runtime/regexp/RegExpScanner.java | 2 +- nashorn/test/script/basic/JDK-8013873.js | 40 +++++++++++++++++++ .../test/script/basic/JDK-8013873.js.EXPECTED | 7 ++++ 3 files changed, 48 insertions(+), 1 deletion(-) create mode 100644 nashorn/test/script/basic/JDK-8013873.js create mode 100644 nashorn/test/script/basic/JDK-8013873.js.EXPECTED diff --git a/nashorn/src/jdk/nashorn/internal/runtime/regexp/RegExpScanner.java b/nashorn/src/jdk/nashorn/internal/runtime/regexp/RegExpScanner.java index 31ee37b174e..956a99d7cf4 100644 --- a/nashorn/src/jdk/nashorn/internal/runtime/regexp/RegExpScanner.java +++ b/nashorn/src/jdk/nashorn/internal/runtime/regexp/RegExpScanner.java @@ -65,7 +65,7 @@ final class RegExpScanner extends Scanner { /** Are we currently inside a negated character class? */ private boolean inNegativeClass = false; - private static final String NON_IDENT_ESCAPES = "$^*+(){}[]|\\.?"; + private static final String NON_IDENT_ESCAPES = "$^*+(){}[]|\\.?-"; private static class Capture { /** diff --git a/nashorn/test/script/basic/JDK-8013873.js b/nashorn/test/script/basic/JDK-8013873.js new file mode 100644 index 00000000000..dcae75cc544 --- /dev/null +++ b/nashorn/test/script/basic/JDK-8013873.js @@ -0,0 +1,40 @@ +/* + * Copyright (c) 2010, 2013, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/** + * JDK-8013873: Regexp regression for escaped dash in character class + * + * @test + * @run + */ + +var re = /[a\-c]/; +print(re.exec("a")); +print(re.exec("-")); +print(re.exec("c")); +print(re.exec("b")); +print(re.exec("\\")); + +re = /^\-$/; +print(re.exec("-")); +print(re.exec("\\-")); diff --git a/nashorn/test/script/basic/JDK-8013873.js.EXPECTED b/nashorn/test/script/basic/JDK-8013873.js.EXPECTED new file mode 100644 index 00000000000..713199368cc --- /dev/null +++ b/nashorn/test/script/basic/JDK-8013873.js.EXPECTED @@ -0,0 +1,7 @@ +a +- +c +null +null +- +null From 5504a220f742b6c1fbe36ec4ee0622f3545ff141 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Hannes=20Walln=C3=B6fer?= Date: Fri, 3 May 2013 22:50:51 +0200 Subject: [PATCH 09/15] 8013874: Function argument's prototype seem cached and wrongly reused Reviewed-by: jlaskey --- .../nashorn/internal/runtime/PropertyMap.java | 7 +- nashorn/test/script/basic/JDK-8013874.js | 66 +++++++++++++++++++ .../test/script/basic/JDK-8013874.js.EXPECTED | 4 ++ 3 files changed, 71 insertions(+), 6 deletions(-) create mode 100644 nashorn/test/script/basic/JDK-8013874.js create mode 100644 nashorn/test/script/basic/JDK-8013874.js.EXPECTED diff --git a/nashorn/src/jdk/nashorn/internal/runtime/PropertyMap.java b/nashorn/src/jdk/nashorn/internal/runtime/PropertyMap.java index c01ceae0d18..077218d3f4f 100644 --- a/nashorn/src/jdk/nashorn/internal/runtime/PropertyMap.java +++ b/nashorn/src/jdk/nashorn/internal/runtime/PropertyMap.java @@ -744,12 +744,7 @@ public final class PropertyMap implements Iterable, PropertyListener { * @return New {@link PropertyMap} with prototype changed. */ PropertyMap changeProto(final ScriptObject oldProto, final ScriptObject newProto) { - if ((oldProto == newProto) || - (size() == 0 && - oldProto == null && - protoGetSwitches == null && - history == null && - protoHistory == null)) { + if (oldProto == newProto) { return this; } diff --git a/nashorn/test/script/basic/JDK-8013874.js b/nashorn/test/script/basic/JDK-8013874.js new file mode 100644 index 00000000000..4b02b92b20a --- /dev/null +++ b/nashorn/test/script/basic/JDK-8013874.js @@ -0,0 +1,66 @@ +/* + * Copyright (c) 2010, 2013, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/** + * JDK-8013874: Function argument's prototype seem cached and wrongly reused + * + * @test + * @run + */ + +function deepEqual(actual, expected) { + print("deepEqual: " + (actual.prop === expected.prop) + ", prop: " + expected.prop); +} + +var proto = {}; +var other = { + prop: 0 +}; + +function NameBuilder(first, last) { + this.first = first; + this.last = last; + return this; +} + +NameBuilder.prototype = proto; + +function NameBuilder2(first, last) { + this.first = first; + this.last = last; + return this; +} + +NameBuilder2.prototype = proto; + +var nb1 = new NameBuilder('Ryan', 'Dahl'); +var nb2 = new NameBuilder2('Ryan', 'Dahl'); + +print("In loader, nb1.prop === nb2.prop " + (nb1.prop === nb2.prop)); +deepEqual(nb1, nb2); + +NameBuilder2.prototype = other; +nb2 = new NameBuilder2('Ryan', 'Dahl'); + +print("In loader, nb1.prop === nb2.prop " + (nb1.prop === nb2.prop)); +deepEqual(nb1, nb2); diff --git a/nashorn/test/script/basic/JDK-8013874.js.EXPECTED b/nashorn/test/script/basic/JDK-8013874.js.EXPECTED new file mode 100644 index 00000000000..af1d1d692e1 --- /dev/null +++ b/nashorn/test/script/basic/JDK-8013874.js.EXPECTED @@ -0,0 +1,4 @@ +In loader, nb1.prop === nb2.prop true +deepEqual: true, prop: undefined +In loader, nb1.prop === nb2.prop false +deepEqual: false, prop: 0 From a788347b73dcbaaba6ee338601490101d5f6af19 Mon Sep 17 00:00:00 2001 From: Marcus Lagergren Date: Tue, 7 May 2013 14:36:57 +0200 Subject: [PATCH 10/15] 8013913: Removed Source field from all nodes except FunctionNode in order to save footprint Reviewed-by: jlaskey, attila --- .../api/scripting/NashornScriptEngine.java | 3 +- .../jdk/nashorn/internal/codegen/Attr.java | 3 +- .../internal/codegen/CodeGenerator.java | 13 +- .../internal/codegen/FinalizeTypes.java | 16 +-- .../internal/codegen/FoldConstants.java | 51 ++++--- .../jdk/nashorn/internal/codegen/Lower.java | 42 +++--- .../nashorn/internal/codegen/Splitter.java | 4 +- .../jdk/nashorn/internal/ir/AccessNode.java | 6 +- .../src/jdk/nashorn/internal/ir/BaseNode.java | 6 +- .../jdk/nashorn/internal/ir/BinaryNode.java | 6 +- .../src/jdk/nashorn/internal/ir/Block.java | 11 +- .../jdk/nashorn/internal/ir/BreakNode.java | 6 +- .../nashorn/internal/ir/BreakableNode.java | 6 +- .../src/jdk/nashorn/internal/ir/CallNode.java | 6 +- .../src/jdk/nashorn/internal/ir/CaseNode.java | 6 +- .../jdk/nashorn/internal/ir/CatchNode.java | 8 +- .../jdk/nashorn/internal/ir/ContinueNode.java | 6 +- .../jdk/nashorn/internal/ir/EmptyNode.java | 6 +- .../jdk/nashorn/internal/ir/ExecuteNode.java | 8 +- .../src/jdk/nashorn/internal/ir/ForNode.java | 6 +- .../jdk/nashorn/internal/ir/FunctionNode.java | 17 ++- .../jdk/nashorn/internal/ir/IdentNode.java | 6 +- .../src/jdk/nashorn/internal/ir/IfNode.java | 6 +- .../jdk/nashorn/internal/ir/IndexNode.java | 6 +- .../jdk/nashorn/internal/ir/LabelNode.java | 6 +- .../nashorn/internal/ir/LexicalContext.java | 18 +-- .../internal/ir/LexicalContextNode.java | 6 +- .../nashorn/internal/ir/LineNumberNode.java | 6 +- .../jdk/nashorn/internal/ir/LiteralNode.java | 102 ++++++------- .../src/jdk/nashorn/internal/ir/Location.java | 134 ------------------ .../src/jdk/nashorn/internal/ir/LoopNode.java | 6 +- nashorn/src/jdk/nashorn/internal/ir/Node.java | 87 ++++++++++-- .../jdk/nashorn/internal/ir/ObjectNode.java | 6 +- .../jdk/nashorn/internal/ir/PropertyNode.java | 6 +- .../jdk/nashorn/internal/ir/ReturnNode.java | 6 +- .../jdk/nashorn/internal/ir/RuntimeNode.java | 11 +- .../jdk/nashorn/internal/ir/SplitNode.java | 2 +- .../jdk/nashorn/internal/ir/SwitchNode.java | 6 +- .../jdk/nashorn/internal/ir/TernaryNode.java | 6 +- .../jdk/nashorn/internal/ir/ThrowNode.java | 7 +- .../src/jdk/nashorn/internal/ir/TryNode.java | 6 +- .../jdk/nashorn/internal/ir/UnaryNode.java | 12 +- .../src/jdk/nashorn/internal/ir/VarNode.java | 11 +- .../jdk/nashorn/internal/ir/WhileNode.java | 6 +- .../src/jdk/nashorn/internal/ir/WithNode.java | 8 +- .../nashorn/internal/ir/debug/JSONWriter.java | 2 +- .../nashorn/internal/objects/NativeDebug.java | 1 - .../internal/parser/AbstractParser.java | 14 +- .../nashorn/internal/parser/JSONParser.java | 14 +- .../jdk/nashorn/internal/parser/Parser.java | 125 ++++++++-------- .../RecompilableScriptFunctionData.java | 4 + .../internal/runtime/ScriptObject.java | 2 - .../runtime/arrays/ArrayLikeIterator.java | 1 - .../runtime/linker/LinkerCallSite.java | 5 +- nashorn/src/jdk/nashorn/tools/Shell.java | 1 - 55 files changed, 366 insertions(+), 515 deletions(-) delete mode 100644 nashorn/src/jdk/nashorn/internal/ir/Location.java diff --git a/nashorn/src/jdk/nashorn/api/scripting/NashornScriptEngine.java b/nashorn/src/jdk/nashorn/api/scripting/NashornScriptEngine.java index 197a6da472f..283e36e9e33 100644 --- a/nashorn/src/jdk/nashorn/api/scripting/NashornScriptEngine.java +++ b/nashorn/src/jdk/nashorn/api/scripting/NashornScriptEngine.java @@ -78,7 +78,6 @@ public final class NashornScriptEngine extends AbstractScriptEngine implements C this(factory, DEFAULT_OPTIONS, appLoader); } - @SuppressWarnings("LeakingThisInConstructor") NashornScriptEngine(final NashornScriptEngineFactory factory, final String[] args, final ClassLoader appLoader) { this.factory = factory; final Options options = new Options("nashorn"); @@ -102,7 +101,7 @@ public final class NashornScriptEngine extends AbstractScriptEngine implements C }); // create new global object - this.global = createNashornGlobal(); + this.global = createNashornGlobal(); // set the default engine scope for the default context context.setBindings(new ScriptObjectMirror(global, global), ScriptContext.ENGINE_SCOPE); diff --git a/nashorn/src/jdk/nashorn/internal/codegen/Attr.java b/nashorn/src/jdk/nashorn/internal/codegen/Attr.java index d5c9ef07a60..a381e58d7aa 100644 --- a/nashorn/src/jdk/nashorn/internal/codegen/Attr.java +++ b/nashorn/src/jdk/nashorn/internal/codegen/Attr.java @@ -435,7 +435,6 @@ final class Attr extends NodeOperatorVisitor { final IdentNode callee = compilerConstant(CALLEE); VarNode selfInit = new VarNode( - newFunctionNode.getSource(), newFunctionNode.getToken(), newFunctionNode.getFinish(), newFunctionNode.getIdent(), @@ -531,6 +530,7 @@ final class Attr extends NodeOperatorVisitor { setBlockScope(name, symbol); if (symbol != null && !identNode.isInitializedHere()) { + symbol.increaseUseCount(); } addLocalUse(identNode.getName()); @@ -914,7 +914,6 @@ final class Attr extends NodeOperatorVisitor { final FunctionNode functionNode = getLexicalContext().getCurrentFunction(); return (IdentNode) new IdentNode( - functionNode.getSource(), functionNode.getToken(), functionNode.getFinish(), cc.symbolName()). diff --git a/nashorn/src/jdk/nashorn/internal/codegen/CodeGenerator.java b/nashorn/src/jdk/nashorn/internal/codegen/CodeGenerator.java index 8f5b3154d25..97219e3db8a 100644 --- a/nashorn/src/jdk/nashorn/internal/codegen/CodeGenerator.java +++ b/nashorn/src/jdk/nashorn/internal/codegen/CodeGenerator.java @@ -261,14 +261,15 @@ final class CodeGenerator extends NodeOperatorVisitor { return method.load(symbol); } - final String name = symbol.getName(); + final String name = symbol.getName(); + final Source source = getLexicalContext().getCurrentFunction().getSource(); if (CompilerConstants.__FILE__.name().equals(name)) { - return method.load(identNode.getSource().getName()); + return method.load(source.getName()); } else if (CompilerConstants.__DIR__.name().equals(name)) { - return method.load(identNode.getSource().getBase()); + return method.load(source.getBase()); } else if (CompilerConstants.__LINE__.name().equals(name)) { - return method.load(identNode.getSource().getLine(identNode.position())).convert(Type.OBJECT); + return method.load(source.getLine(identNode.position())).convert(Type.OBJECT); } else { assert identNode.getSymbol().isScope() : identNode + " is not in scope!"; @@ -2005,8 +2006,9 @@ final class CodeGenerator extends NodeOperatorVisitor { public boolean enterThrowNode(final ThrowNode throwNode) { method._new(ECMAException.class).dup(); + final Source source = getLexicalContext().getCurrentFunction().getSource(); + final Node expression = throwNode.getExpression(); - final Source source = throwNode.getSource(); final int position = throwNode.position(); final int line = source.getLine(position); final int column = source.getColumn(position); @@ -3013,7 +3015,6 @@ final class CodeGenerator extends NodeOperatorVisitor { return; } - @SuppressWarnings("resource") final PrintWriter out = compiler.getEnv().getErr(); out.println("[BLOCK in '" + ident + "']"); if (!block.printSymbols(out)) { diff --git a/nashorn/src/jdk/nashorn/internal/codegen/FinalizeTypes.java b/nashorn/src/jdk/nashorn/internal/codegen/FinalizeTypes.java index 895b344e3e8..1b6e3c161df 100644 --- a/nashorn/src/jdk/nashorn/internal/codegen/FinalizeTypes.java +++ b/nashorn/src/jdk/nashorn/internal/codegen/FinalizeTypes.java @@ -773,7 +773,7 @@ final class FinalizeTypes extends NodeOperatorVisitor { private Node convert(final Node node, final Type to) { assert !to.isUnknown() : "unknown type for " + node + " class=" + node.getClass(); assert node != null : "node is null"; - assert node.getSymbol() != null : "node " + node + " " + node.getClass() + " has no symbol! " + getLexicalContext().getCurrentFunction() + " " + node.getSource(); + assert node.getSymbol() != null : "node " + node + " " + node.getClass() + " has no symbol! " + getLexicalContext().getCurrentFunction(); assert node.tokenType() != TokenType.CONVERT : "assert convert in convert " + node + " in " + getLexicalContext().getCurrentFunction(); final Type from = node.getType(); @@ -798,7 +798,7 @@ final class FinalizeTypes extends NodeOperatorVisitor { assert node instanceof TypeOverride; return setTypeOverride(node, to); } - resultNode = new UnaryNode(node.getSource(), Token.recast(node.getToken(), TokenType.CONVERT), node); + resultNode = new UnaryNode(Token.recast(node.getToken(), TokenType.CONVERT), node); } LOG.info("CONVERT('", node, "', ", to, ") => '", resultNode, "'"); @@ -813,7 +813,7 @@ final class FinalizeTypes extends NodeOperatorVisitor { private static Node discard(final Node node) { if (node.getSymbol() != null) { - final Node discard = new UnaryNode(node.getSource(), Token.recast(node.getToken(), TokenType.DISCARD), node); + final Node discard = new UnaryNode(Token.recast(node.getToken(), TokenType.DISCARD), node); //discard never has a symbol in the discard node - then it would be a nop assert !node.isTerminal(); return discard; @@ -881,15 +881,15 @@ final class FinalizeTypes extends NodeOperatorVisitor { LiteralNode literalNode = null; if (type.isString()) { - literalNode = LiteralNode.newInstance(source, token, finish, JSType.toString(value)); + literalNode = LiteralNode.newInstance(token, finish, JSType.toString(value)); } else if (type.isBoolean()) { - literalNode = LiteralNode.newInstance(source, token, finish, JSType.toBoolean(value)); + literalNode = LiteralNode.newInstance(token, finish, JSType.toBoolean(value)); } else if (type.isInteger()) { - literalNode = LiteralNode.newInstance(source, token, finish, JSType.toInt32(value)); + literalNode = LiteralNode.newInstance(token, finish, JSType.toInt32(value)); } else if (type.isLong()) { - literalNode = LiteralNode.newInstance(source, token, finish, JSType.toLong(value)); + literalNode = LiteralNode.newInstance(token, finish, JSType.toLong(value)); } else if (type.isNumber() || parent.getType().isNumeric() && !parent.getType().isNumber()) { - literalNode = LiteralNode.newInstance(source, token, finish, JSType.toNumber(value)); + literalNode = LiteralNode.newInstance(token, finish, JSType.toNumber(value)); } if (literalNode != null) { diff --git a/nashorn/src/jdk/nashorn/internal/codegen/FoldConstants.java b/nashorn/src/jdk/nashorn/internal/codegen/FoldConstants.java index 03accb4b517..92286c2bf63 100644 --- a/nashorn/src/jdk/nashorn/internal/codegen/FoldConstants.java +++ b/nashorn/src/jdk/nashorn/internal/codegen/FoldConstants.java @@ -41,7 +41,6 @@ import jdk.nashorn.internal.ir.visitor.NodeVisitor; import jdk.nashorn.internal.runtime.DebugLogger; import jdk.nashorn.internal.runtime.JSType; import jdk.nashorn.internal.runtime.ScriptRuntime; -import jdk.nashorn.internal.runtime.Source; /** * Simple constant folding pass, executed before IR is starting to be lowered. @@ -112,13 +111,11 @@ final class FoldConstants extends NodeVisitor { */ abstract static class ConstantEvaluator { protected T parent; - protected final Source source; protected final long token; protected final int finish; protected ConstantEvaluator(final T parent) { this.parent = parent; - this.source = parent.getSource(); this.token = parent.getToken(); this.finish = parent.getFinish(); } @@ -152,23 +149,23 @@ final class FoldConstants extends NodeVisitor { switch (parent.tokenType()) { case ADD: if (rhsInteger) { - literalNode = LiteralNode.newInstance(source, token, finish, rhs.getInt32()); + literalNode = LiteralNode.newInstance(token, finish, rhs.getInt32()); } else { - literalNode = LiteralNode.newInstance(source, token, finish, rhs.getNumber()); + literalNode = LiteralNode.newInstance(token, finish, rhs.getNumber()); } break; case SUB: if (rhsInteger && rhs.getInt32() != 0) { // @see test/script/basic/minuszero.js - literalNode = LiteralNode.newInstance(source, token, finish, -rhs.getInt32()); + literalNode = LiteralNode.newInstance(token, finish, -rhs.getInt32()); } else { - literalNode = LiteralNode.newInstance(source, token, finish, -rhs.getNumber()); + literalNode = LiteralNode.newInstance(token, finish, -rhs.getNumber()); } break; case NOT: - literalNode = LiteralNode.newInstance(source, token, finish, !rhs.getBoolean()); + literalNode = LiteralNode.newInstance(token, finish, !rhs.getBoolean()); break; case BIT_NOT: - literalNode = LiteralNode.newInstance(source, token, finish, ~rhs.getInt32()); + literalNode = LiteralNode.newInstance(token, finish, ~rhs.getInt32()); break; default: return null; @@ -234,7 +231,7 @@ final class FoldConstants extends NodeVisitor { break; } assert res instanceof CharSequence : res + " was not a CharSequence, it was a " + res.getClass(); - return LiteralNode.newInstance(source, token, finish, res.toString()); + return LiteralNode.newInstance(token, finish, res.toString()); } return null; case MUL: @@ -247,33 +244,33 @@ final class FoldConstants extends NodeVisitor { value = lhs.getNumber() - rhs.getNumber(); break; case SHR: - return LiteralNode.newInstance(source, token, finish, (lhs.getInt32() >>> rhs.getInt32()) & JSType.MAX_UINT); + return LiteralNode.newInstance(token, finish, (lhs.getInt32() >>> rhs.getInt32()) & JSType.MAX_UINT); case SAR: - return LiteralNode.newInstance(source, token, finish, lhs.getInt32() >> rhs.getInt32()); + return LiteralNode.newInstance(token, finish, lhs.getInt32() >> rhs.getInt32()); case SHL: - return LiteralNode.newInstance(source, token, finish, lhs.getInt32() << rhs.getInt32()); + return LiteralNode.newInstance(token, finish, lhs.getInt32() << rhs.getInt32()); case BIT_XOR: - return LiteralNode.newInstance(source, token, finish, lhs.getInt32() ^ rhs.getInt32()); + return LiteralNode.newInstance(token, finish, lhs.getInt32() ^ rhs.getInt32()); case BIT_AND: - return LiteralNode.newInstance(source, token, finish, lhs.getInt32() & rhs.getInt32()); + return LiteralNode.newInstance(token, finish, lhs.getInt32() & rhs.getInt32()); case BIT_OR: - return LiteralNode.newInstance(source, token, finish, lhs.getInt32() | rhs.getInt32()); + return LiteralNode.newInstance(token, finish, lhs.getInt32() | rhs.getInt32()); case GE: - return LiteralNode.newInstance(source, token, finish, ScriptRuntime.GE(lhs.getObject(), rhs.getObject())); + return LiteralNode.newInstance(token, finish, ScriptRuntime.GE(lhs.getObject(), rhs.getObject())); case LE: - return LiteralNode.newInstance(source, token, finish, ScriptRuntime.LE(lhs.getObject(), rhs.getObject())); + return LiteralNode.newInstance(token, finish, ScriptRuntime.LE(lhs.getObject(), rhs.getObject())); case GT: - return LiteralNode.newInstance(source, token, finish, ScriptRuntime.GT(lhs.getObject(), rhs.getObject())); + return LiteralNode.newInstance(token, finish, ScriptRuntime.GT(lhs.getObject(), rhs.getObject())); case LT: - return LiteralNode.newInstance(source, token, finish, ScriptRuntime.LT(lhs.getObject(), rhs.getObject())); + return LiteralNode.newInstance(token, finish, ScriptRuntime.LT(lhs.getObject(), rhs.getObject())); case NE: - return LiteralNode.newInstance(source, token, finish, ScriptRuntime.NE(lhs.getObject(), rhs.getObject())); + return LiteralNode.newInstance(token, finish, ScriptRuntime.NE(lhs.getObject(), rhs.getObject())); case NE_STRICT: - return LiteralNode.newInstance(source, token, finish, ScriptRuntime.NE_STRICT(lhs.getObject(), rhs.getObject())); + return LiteralNode.newInstance(token, finish, ScriptRuntime.NE_STRICT(lhs.getObject(), rhs.getObject())); case EQ: - return LiteralNode.newInstance(source, token, finish, ScriptRuntime.EQ(lhs.getObject(), rhs.getObject())); + return LiteralNode.newInstance(token, finish, ScriptRuntime.EQ(lhs.getObject(), rhs.getObject())); case EQ_STRICT: - return LiteralNode.newInstance(source, token, finish, ScriptRuntime.EQ_STRICT(lhs.getObject(), rhs.getObject())); + return LiteralNode.newInstance(token, finish, ScriptRuntime.EQ_STRICT(lhs.getObject(), rhs.getObject())); default: return null; } @@ -282,12 +279,12 @@ final class FoldConstants extends NodeVisitor { isLong &= value != 0.0 && JSType.isRepresentableAsLong(value); if (isInteger) { - return LiteralNode.newInstance(source, token, finish, JSType.toInt32(value)); + return LiteralNode.newInstance(token, finish, JSType.toInt32(value)); } else if (isLong) { - return LiteralNode.newInstance(source, token, finish, JSType.toLong(value)); + return LiteralNode.newInstance(token, finish, JSType.toLong(value)); } - return LiteralNode.newInstance(source, token, finish, value); + return LiteralNode.newInstance(token, finish, value); } } } diff --git a/nashorn/src/jdk/nashorn/internal/codegen/Lower.java b/nashorn/src/jdk/nashorn/internal/codegen/Lower.java index d6209e4fd01..6f55841ebc9 100644 --- a/nashorn/src/jdk/nashorn/internal/codegen/Lower.java +++ b/nashorn/src/jdk/nashorn/internal/codegen/Lower.java @@ -118,8 +118,9 @@ final class Lower extends NodeOperatorVisitor { @Override public boolean enterBlock(final Block block) { final LexicalContext lc = getLexicalContext(); - if (lc.isFunctionBody() && lc.getCurrentFunction().isProgram() && !lc.getCurrentFunction().hasDeclaredFunctions()) { - new ExecuteNode(block.getSource(), block.getToken(), block.getFinish(), LiteralNode.newInstance(block, ScriptRuntime.UNDEFINED)).accept(this); + final FunctionNode function = lc.getCurrentFunction(); + if (lc.isFunctionBody() && function.isProgram() && !function.hasDeclaredFunctions()) { + new ExecuteNode(block.getToken(), block.getFinish(), LiteralNode.newInstance(block, ScriptRuntime.UNDEFINED)).accept(this); } return true; } @@ -137,7 +138,6 @@ final class Lower extends NodeOperatorVisitor { final FunctionNode currentFunction = getLexicalContext().getCurrentFunction(); final boolean isProgram = currentFunction.isProgram(); final ReturnNode returnNode = new ReturnNode( - currentFunction.getSource(), currentFunction.getToken(), currentFunction.getFinish(), isProgram ? @@ -193,7 +193,6 @@ final class Lower extends NodeOperatorVisitor { if (!isInternalExpression(expr) && !isEvalResultAssignment(expr)) { node = executeNode.setExpression( new BinaryNode( - executeNode.getSource(), Token.recast( executeNode.getToken(), TokenType.ASSIGN), @@ -284,17 +283,16 @@ final class Lower extends NodeOperatorVisitor { } private Block catchAllBlock(final TryNode tryNode) { - final Source source = tryNode.getSource(); final long token = tryNode.getToken(); final int finish = tryNode.getFinish(); - final IdentNode exception = new IdentNode(source, token, finish, getLexicalContext().getCurrentFunction().uniqueName("catch_all")); + final IdentNode exception = new IdentNode(token, finish, getLexicalContext().getCurrentFunction().uniqueName("catch_all")); - final Block catchBody = new Block(source, token, finish, new ThrowNode(source, token, finish, new IdentNode(exception))). + final Block catchBody = new Block(token, finish, new ThrowNode(token, finish, new IdentNode(exception))). setIsTerminal(getLexicalContext(), true); //ends with throw, so terminal - final CatchNode catchAllNode = new CatchNode(source, token, finish, new IdentNode(exception), null, catchBody); - final Block catchAllBlock = new Block(source, token, finish, catchAllNode); + final CatchNode catchAllNode = new CatchNode(token, finish, new IdentNode(exception), null, catchBody); + final Block catchAllBlock = new Block(token, finish, catchAllNode); //catchallblock -> catchallnode (catchnode) -> exception -> throw @@ -303,7 +301,7 @@ final class Lower extends NodeOperatorVisitor { private IdentNode compilerConstant(final CompilerConstants cc) { final FunctionNode functionNode = getLexicalContext().getCurrentFunction(); - return new IdentNode(functionNode.getSource(), functionNode.getToken(), functionNode.getFinish(), cc.symbolName()); + return new IdentNode(functionNode.getToken(), functionNode.getFinish(), cc.symbolName()); } private static boolean isTerminal(final List statements) { @@ -318,7 +316,6 @@ final class Lower extends NodeOperatorVisitor { * @return new try node after splicing finally code (same if nop) */ private Node spliceFinally(final TryNode tryNode, final List rethrows, final Block finallyBody) { - final Source source = tryNode.getSource(); final int finish = tryNode.getFinish(); assert tryNode.getFinallyBody() == null; @@ -345,7 +342,7 @@ final class Lower extends NodeOperatorVisitor { if (!isTerminal(newStatements)) { newStatements.add(throwNode); } - return new Block(source, throwNode.getToken(), throwNode.getFinish(), newStatements); + return new Block(throwNode.getToken(), throwNode.getFinish(), newStatements); } return throwNode; } @@ -370,7 +367,7 @@ final class Lower extends NodeOperatorVisitor { //we need to evaluate the result of the return in case it is complex while //still in the try block, store it in a result value and return it afterwards resultNode = new IdentNode(Lower.this.compilerConstant(RETURN)); - newStatements.add(new ExecuteNode(new BinaryNode(source, Token.recast(returnNode.getToken(), TokenType.ASSIGN), resultNode, expr))); + newStatements.add(new ExecuteNode(new BinaryNode(Token.recast(returnNode.getToken(), TokenType.ASSIGN), resultNode, expr))); } else { resultNode = null; } @@ -380,7 +377,7 @@ final class Lower extends NodeOperatorVisitor { newStatements.add(expr == null ? returnNode : returnNode.setExpression(resultNode)); } - return new ExecuteNode(new Block(source, returnNode.getToken(), getLexicalContext().getCurrentBlock().getFinish(), newStatements)); + return new ExecuteNode(new Block(returnNode.getToken(), getLexicalContext().getCurrentBlock().getFinish(), newStatements)); } private Node copy(final Node endpoint, final Node targetNode) { @@ -389,7 +386,7 @@ final class Lower extends NodeOperatorVisitor { if (!isTerminal(newStatements)) { newStatements.add(endpoint); } - return new ExecuteNode(new Block(source, endpoint.getToken(), finish, newStatements)); + return new ExecuteNode(new Block(endpoint.getToken(), finish, newStatements)); } return endpoint; } @@ -451,7 +448,7 @@ final class Lower extends NodeOperatorVisitor { if (tryNode.getCatchBlocks().isEmpty()) { newTryNode = tryNode.setFinallyBody(null); } else { - Block outerBody = new Block(tryNode.getSource(), tryNode.getToken(), tryNode.getFinish(), new ArrayList(Arrays.asList(tryNode.setFinallyBody(null)))); + Block outerBody = new Block(tryNode.getToken(), tryNode.getFinish(), new ArrayList(Arrays.asList(tryNode.setFinallyBody(null)))); newTryNode = tryNode.setBody(outerBody).setCatchBlocks(null); } @@ -468,19 +465,19 @@ final class Lower extends NodeOperatorVisitor { public Node leaveVarNode(final VarNode varNode) { addStatement(varNode); if (varNode.getFlag(VarNode.IS_LAST_FUNCTION_DECLARATION) && getLexicalContext().getCurrentFunction().isProgram()) { - new ExecuteNode(varNode.getSource(), varNode.getToken(), varNode.getFinish(), new IdentNode(varNode.getName())).accept(this); + new ExecuteNode(varNode.getToken(), varNode.getFinish(), new IdentNode(varNode.getName())).accept(this); } return varNode; } @Override public Node leaveWhileNode(final WhileNode whileNode) { - final Node test = whileNode.getTest(); + final Node test = whileNode.getTest(); final Block body = whileNode.getBody(); if (conservativeAlwaysTrue(test)) { //turn it into a for node without a test. - final ForNode forNode = (ForNode)new ForNode(whileNode.getSource(), whileNode.getToken(), whileNode.getFinish(), null, null, body, null, ForNode.IS_FOR).accept(this); + final ForNode forNode = (ForNode)new ForNode(whileNode.getToken(), whileNode.getFinish(), null, null, body, null, ForNode.IS_FOR).accept(this); getLexicalContext().replace(whileNode, forNode); return forNode; } @@ -525,11 +522,12 @@ final class Lower extends NodeOperatorVisitor { * @param node a node * @return eval location */ - private static String evalLocation(final IdentNode node) { + private String evalLocation(final IdentNode node) { + final Source source = getLexicalContext().getCurrentFunction().getSource(); return new StringBuilder(). - append(node.getSource().getName()). + append(source.getName()). append('#'). - append(node.getSource().getLine(node.position())). + append(source.getLine(node.position())). append(""). toString(); } diff --git a/nashorn/src/jdk/nashorn/internal/codegen/Splitter.java b/nashorn/src/jdk/nashorn/internal/codegen/Splitter.java index 8049473c87b..e86a516e0f8 100644 --- a/nashorn/src/jdk/nashorn/internal/codegen/Splitter.java +++ b/nashorn/src/jdk/nashorn/internal/codegen/Splitter.java @@ -42,7 +42,6 @@ import jdk.nashorn.internal.ir.Node; import jdk.nashorn.internal.ir.SplitNode; import jdk.nashorn.internal.ir.visitor.NodeVisitor; import jdk.nashorn.internal.runtime.DebugLogger; -import jdk.nashorn.internal.runtime.Source; import jdk.nashorn.internal.runtime.options.Options; /** @@ -221,12 +220,11 @@ final class Splitter extends NodeVisitor { * @return New split node. */ private SplitNode createBlockSplitNode(final Block parent, final FunctionNode function, final List statements, final long weight) { - final Source source = parent.getSource(); final long token = parent.getToken(); final int finish = parent.getFinish(); final String name = function.uniqueName(SPLIT_PREFIX.symbolName()); - final Block newBlock = new Block(source, token, finish, statements); + final Block newBlock = new Block(token, finish, statements); return new SplitNode(name, newBlock, compiler.findUnit(weight + WeighNodes.FUNCTION_WEIGHT)); } diff --git a/nashorn/src/jdk/nashorn/internal/ir/AccessNode.java b/nashorn/src/jdk/nashorn/internal/ir/AccessNode.java index 2f739bf03a1..a179056598a 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/AccessNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/AccessNode.java @@ -28,7 +28,6 @@ package jdk.nashorn.internal.ir; import jdk.nashorn.internal.codegen.types.Type; import jdk.nashorn.internal.ir.annotations.Immutable; import jdk.nashorn.internal.ir.visitor.NodeVisitor; -import jdk.nashorn.internal.runtime.Source; /** * IR representation of a property access (period operator.) @@ -41,14 +40,13 @@ public final class AccessNode extends BaseNode { /** * Constructor * - * @param source source code * @param token token * @param finish finish * @param base base node * @param property property */ - public AccessNode(final Source source, final long token, final int finish, final Node base, final IdentNode property) { - super(source, token, finish, base, false, false); + public AccessNode(final long token, final int finish, final Node base, final IdentNode property) { + super(token, finish, base, false, false); this.property = property.setIsPropertyName(); } diff --git a/nashorn/src/jdk/nashorn/internal/ir/BaseNode.java b/nashorn/src/jdk/nashorn/internal/ir/BaseNode.java index 5e5bfb13c59..a1b7c0eed7d 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/BaseNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/BaseNode.java @@ -29,7 +29,6 @@ import static jdk.nashorn.internal.codegen.ObjectClassGenerator.DEBUG_FIELDS; import jdk.nashorn.internal.codegen.ObjectClassGenerator; import jdk.nashorn.internal.codegen.types.Type; import jdk.nashorn.internal.ir.annotations.Immutable; -import jdk.nashorn.internal.runtime.Source; /** * IR base for accessing/indexing nodes. @@ -50,15 +49,14 @@ public abstract class BaseNode extends Node implements FunctionCall, TypeOverrid /** * Constructor * - * @param source source code * @param token token * @param finish finish * @param base base node * @param isFunction is this a function * @param hasCallSiteType does this access have a callsite type */ - public BaseNode(final Source source, final long token, final int finish, final Node base, final boolean isFunction, final boolean hasCallSiteType) { - super(source, token, base.getStart(), finish); + public BaseNode(final long token, final int finish, final Node base, final boolean isFunction, final boolean hasCallSiteType) { + super(token, base.getStart(), finish); this.base = base; this.isFunction = isFunction; this.hasCallSiteType = hasCallSiteType; diff --git a/nashorn/src/jdk/nashorn/internal/ir/BinaryNode.java b/nashorn/src/jdk/nashorn/internal/ir/BinaryNode.java index 28df8edfcc8..61454fef0a1 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/BinaryNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/BinaryNode.java @@ -29,7 +29,6 @@ import jdk.nashorn.internal.codegen.types.Type; import jdk.nashorn.internal.ir.annotations.Immutable; import jdk.nashorn.internal.ir.visitor.NodeVisitor; import jdk.nashorn.internal.parser.TokenType; -import jdk.nashorn.internal.runtime.Source; /** * BinaryNode nodes represent two operand operations. @@ -44,13 +43,12 @@ public final class BinaryNode extends Node implements Assignment { /** * Constructor * - * @param source source code * @param token token * @param lhs left hand side * @param rhs right hand side */ - public BinaryNode(final Source source, final long token, final Node lhs, final Node rhs) { - super(source, token, lhs.getStart(), rhs.getFinish()); + public BinaryNode(final long token, final Node lhs, final Node rhs) { + super(token, lhs.getStart(), rhs.getFinish()); this.lhs = lhs; this.rhs = rhs; } diff --git a/nashorn/src/jdk/nashorn/internal/ir/Block.java b/nashorn/src/jdk/nashorn/internal/ir/Block.java index 15a00869f0a..9e5d69a793d 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/Block.java +++ b/nashorn/src/jdk/nashorn/internal/ir/Block.java @@ -36,7 +36,6 @@ import java.util.Map; import jdk.nashorn.internal.codegen.Label; import jdk.nashorn.internal.ir.annotations.Immutable; import jdk.nashorn.internal.ir.visitor.NodeVisitor; -import jdk.nashorn.internal.runtime.Source; /** * IR representation for a list of statements and functions. All provides the @@ -77,13 +76,12 @@ public class Block extends BreakableNode implements Flags { /** * Constructor * - * @param source source code * @param token token * @param finish finish * @param statements statements */ - public Block(final Source source, final long token, final int finish, final Node... statements) { - super(source, token, finish, new Label("block_break")); + public Block(final long token, final int finish, final Node... statements) { + super(token, finish, new Label("block_break")); this.statements = Arrays.asList(statements); this.symbols = new LinkedHashMap<>(); @@ -94,13 +92,12 @@ public class Block extends BreakableNode implements Flags { /** * Constructor * - * @param source source code * @param token token * @param finish finish * @param statements statements */ - public Block(final Source source, final long token, final int finish, final List statements) { - this(source, token, finish, statements.toArray(new Node[statements.size()])); + public Block(final long token, final int finish, final List statements) { + this(token, finish, statements.toArray(new Node[statements.size()])); } private Block(final Block block, final int finish, final List statements, final int flags, final Map symbols) { diff --git a/nashorn/src/jdk/nashorn/internal/ir/BreakNode.java b/nashorn/src/jdk/nashorn/internal/ir/BreakNode.java index f0966b4692b..630c1ab619e 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/BreakNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/BreakNode.java @@ -27,7 +27,6 @@ package jdk.nashorn.internal.ir; import jdk.nashorn.internal.ir.annotations.Immutable; import jdk.nashorn.internal.ir.visitor.NodeVisitor; -import jdk.nashorn.internal.runtime.Source; /** * IR representation for {@code break} statements. @@ -40,13 +39,12 @@ public final class BreakNode extends Node { /** * Constructor * - * @param source source code * @param token token * @param finish finish * @param label label for break or null if none */ - public BreakNode(final Source source, final long token, final int finish, final IdentNode label) { - super(source, token, finish); + public BreakNode(final long token, final int finish, final IdentNode label) { + super(token, finish); this.label = label; } diff --git a/nashorn/src/jdk/nashorn/internal/ir/BreakableNode.java b/nashorn/src/jdk/nashorn/internal/ir/BreakableNode.java index 3662bfa774e..f433dc4d75f 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/BreakableNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/BreakableNode.java @@ -30,7 +30,6 @@ import java.util.List; import jdk.nashorn.internal.codegen.Label; import jdk.nashorn.internal.ir.annotations.Immutable; -import jdk.nashorn.internal.runtime.Source; /** * This class represents a node from which control flow can execute @@ -45,13 +44,12 @@ public abstract class BreakableNode extends LexicalContextNode { /** * Constructor * - * @param source source code * @param token token * @param finish finish * @param breakLabel break label */ - protected BreakableNode(final Source source, final long token, final int finish, final Label breakLabel) { - super(source, token, finish); + protected BreakableNode(final long token, final int finish, final Label breakLabel) { + super(token, finish); this.breakLabel = breakLabel; } diff --git a/nashorn/src/jdk/nashorn/internal/ir/CallNode.java b/nashorn/src/jdk/nashorn/internal/ir/CallNode.java index 9a730aa862f..71504abc572 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/CallNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/CallNode.java @@ -31,7 +31,6 @@ import jdk.nashorn.internal.codegen.types.Type; import jdk.nashorn.internal.ir.annotations.Ignore; import jdk.nashorn.internal.ir.annotations.Immutable; import jdk.nashorn.internal.ir.visitor.NodeVisitor; -import jdk.nashorn.internal.runtime.Source; /** * IR representation for a function call. @@ -137,14 +136,13 @@ public final class CallNode extends LexicalContextNode implements TypeOverride args) { - super(source, token, finish); + public CallNode(final long token, final int finish, final Node function, final List args) { + super(token, finish); this.function = function; this.args = args; diff --git a/nashorn/src/jdk/nashorn/internal/ir/CaseNode.java b/nashorn/src/jdk/nashorn/internal/ir/CaseNode.java index 237536cc490..8a438f8fb71 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/CaseNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/CaseNode.java @@ -28,7 +28,6 @@ package jdk.nashorn.internal.ir; import jdk.nashorn.internal.codegen.Label; import jdk.nashorn.internal.ir.annotations.Immutable; import jdk.nashorn.internal.ir.visitor.NodeVisitor; -import jdk.nashorn.internal.runtime.Source; /** * IR representation of CASE clause. @@ -48,14 +47,13 @@ public final class CaseNode extends Node { /** * Constructors * - * @param source the source * @param token token * @param finish finish * @param test case test node, can be any node in JavaScript * @param body case body */ - public CaseNode(final Source source, final long token, final int finish, final Node test, final Block body) { - super(source, token, finish); + public CaseNode(final long token, final int finish, final Node test, final Block body) { + super(token, finish); this.test = test; this.body = body; diff --git a/nashorn/src/jdk/nashorn/internal/ir/CatchNode.java b/nashorn/src/jdk/nashorn/internal/ir/CatchNode.java index e7d231e8d82..993b996a6a2 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/CatchNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/CatchNode.java @@ -27,7 +27,6 @@ package jdk.nashorn.internal.ir; import jdk.nashorn.internal.ir.annotations.Immutable; import jdk.nashorn.internal.ir.visitor.NodeVisitor; -import jdk.nashorn.internal.runtime.Source; /** * IR representation of a catch clause. @@ -46,16 +45,14 @@ public final class CatchNode extends Node { /** * Constructors * - * @param source the source * @param token token * @param finish finish * @param exception variable name of exception * @param exceptionCondition exception condition * @param body catch body */ - public CatchNode(final Source source, final long token, final int finish, final IdentNode exception, final Node exceptionCondition, final Block body) { - super(source, token, finish); - + public CatchNode(final long token, final int finish, final IdentNode exception, final Node exceptionCondition, final Block body) { + super(token, finish); this.exception = exception; this.exceptionCondition = exceptionCondition; this.body = body; @@ -63,7 +60,6 @@ public final class CatchNode extends Node { private CatchNode(final CatchNode catchNode, final IdentNode exception, final Node exceptionCondition, final Block body) { super(catchNode); - this.exception = exception; this.exceptionCondition = exceptionCondition; this.body = body; diff --git a/nashorn/src/jdk/nashorn/internal/ir/ContinueNode.java b/nashorn/src/jdk/nashorn/internal/ir/ContinueNode.java index c8cc309d51e..2abdc7d01a5 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/ContinueNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/ContinueNode.java @@ -27,7 +27,6 @@ package jdk.nashorn.internal.ir; import jdk.nashorn.internal.ir.annotations.Immutable; import jdk.nashorn.internal.ir.visitor.NodeVisitor; -import jdk.nashorn.internal.runtime.Source; /** * IR representation for CONTINUE statements. @@ -40,13 +39,12 @@ public class ContinueNode extends Node { /** * Constructor * - * @param source source code * @param token token * @param finish finish * @param label label for break or null if none */ - public ContinueNode(final Source source, final long token, final int finish, final IdentNode label) { - super(source, token, finish); + public ContinueNode(final long token, final int finish, final IdentNode label) { + super(token, finish); this.label = label; } diff --git a/nashorn/src/jdk/nashorn/internal/ir/EmptyNode.java b/nashorn/src/jdk/nashorn/internal/ir/EmptyNode.java index 67516127057..06a4f66f8db 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/EmptyNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/EmptyNode.java @@ -27,7 +27,6 @@ package jdk.nashorn.internal.ir; import jdk.nashorn.internal.ir.annotations.Immutable; import jdk.nashorn.internal.ir.visitor.NodeVisitor; -import jdk.nashorn.internal.runtime.Source; /** * IR representation for an empty statement. @@ -47,12 +46,11 @@ public final class EmptyNode extends Node { /** * Constructor * - * @param source the source * @param token token * @param finish finish */ - public EmptyNode(final Source source, final long token, final int finish) { - super(source, token, finish); + public EmptyNode(final long token, final int finish) { + super(token, finish); } diff --git a/nashorn/src/jdk/nashorn/internal/ir/ExecuteNode.java b/nashorn/src/jdk/nashorn/internal/ir/ExecuteNode.java index f6dd7d1bf8c..8a9e27dd75e 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/ExecuteNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/ExecuteNode.java @@ -27,7 +27,6 @@ package jdk.nashorn.internal.ir; import jdk.nashorn.internal.ir.annotations.Immutable; import jdk.nashorn.internal.ir.visitor.NodeVisitor; -import jdk.nashorn.internal.runtime.Source; /** * IR representation for executing bare expressions. Basically, an expression @@ -42,13 +41,12 @@ public final class ExecuteNode extends Node { /** * Constructor * - * @param source the source * @param token token * @param finish finish * @param expression the expression to execute */ - public ExecuteNode(final Source source, final long token, final int finish, final Node expression) { - super(source, token, finish); + public ExecuteNode(final long token, final int finish, final Node expression) { + super(token, finish); this.expression = expression; } @@ -63,7 +61,7 @@ public final class ExecuteNode extends Node { * @param expression an expression to wrap, from which source, tokens and finish are also inherited */ public ExecuteNode(final Node expression) { - super(expression.getSource(), expression.getToken(), expression.getFinish()); + super(expression.getToken(), expression.getFinish()); this.expression = expression; } diff --git a/nashorn/src/jdk/nashorn/internal/ir/ForNode.java b/nashorn/src/jdk/nashorn/internal/ir/ForNode.java index 057b8464ac7..17a63a66495 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/ForNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/ForNode.java @@ -27,7 +27,6 @@ package jdk.nashorn.internal.ir; import jdk.nashorn.internal.ir.annotations.Immutable; import jdk.nashorn.internal.ir.visitor.NodeVisitor; -import jdk.nashorn.internal.runtime.Source; /** * IR representing a FOR statement. @@ -57,7 +56,6 @@ public final class ForNode extends LoopNode { /** * Constructor * - * @param source the source * @param token token * @param finish finish * @param init init @@ -66,8 +64,8 @@ public final class ForNode extends LoopNode { * @param modify modify * @param flags flags */ - public ForNode(final Source source, final long token, final int finish, final Node init, final Node test, final Block body, final Node modify, final int flags) { - super(source, token, finish, test, body, false); + public ForNode(final long token, final int finish, final Node init, final Node test, final Block body, final Node modify, final int flags) { + super(token, finish, test, body, false); this.init = init; this.modify = modify; this.flags = flags; diff --git a/nashorn/src/jdk/nashorn/internal/ir/FunctionNode.java b/nashorn/src/jdk/nashorn/internal/ir/FunctionNode.java index dc11771de3b..728d81cbca4 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/FunctionNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/FunctionNode.java @@ -86,6 +86,8 @@ public final class FunctionNode extends LexicalContextNode implements Flags parameters, final FunctionNode.Kind kind, final int flags) { - super(source, token, finish); + super(token, finish); + this.source = source; this.ident = ident; this.name = name; this.kind = kind; @@ -265,6 +268,7 @@ public final class FunctionNode extends LexicalContextNode implements Flags ]"); diff --git a/nashorn/src/jdk/nashorn/internal/ir/LexicalContextNode.java b/nashorn/src/jdk/nashorn/internal/ir/LexicalContextNode.java index c4d78224271..dffdaae8480 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/LexicalContextNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/LexicalContextNode.java @@ -25,7 +25,6 @@ package jdk.nashorn.internal.ir; import jdk.nashorn.internal.ir.visitor.NodeVisitor; -import jdk.nashorn.internal.runtime.Source; /** * Superclass for nodes that can be part of the lexical context @@ -35,12 +34,11 @@ public abstract class LexicalContextNode extends Node { /** * Constructor * - * @param source source * @param token token * @param finish finish */ - protected LexicalContextNode(final Source source, final long token, final int finish) { - super(source, token, finish); + protected LexicalContextNode(final long token, final int finish) { + super(token, finish); } /** diff --git a/nashorn/src/jdk/nashorn/internal/ir/LineNumberNode.java b/nashorn/src/jdk/nashorn/internal/ir/LineNumberNode.java index 63f04594451..59bfb0c564a 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/LineNumberNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/LineNumberNode.java @@ -28,7 +28,6 @@ package jdk.nashorn.internal.ir; import jdk.nashorn.internal.ir.annotations.Immutable; import jdk.nashorn.internal.ir.visitor.NodeVisitor; import jdk.nashorn.internal.parser.Token; -import jdk.nashorn.internal.runtime.Source; /** * IR Node representing a line number @@ -41,12 +40,11 @@ public final class LineNumberNode extends Node { /** * Constructor * - * @param source the source * @param token token * @param lineNumber the line number */ - public LineNumberNode(final Source source, final long token, final int lineNumber) { - super(source, token, Token.descPosition(token)); + public LineNumberNode(final long token, final int lineNumber) { + super(token, Token.descPosition(token)); this.lineNumber = lineNumber; } diff --git a/nashorn/src/jdk/nashorn/internal/ir/LiteralNode.java b/nashorn/src/jdk/nashorn/internal/ir/LiteralNode.java index 5d41c9562d6..4c2f932cf4d 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/LiteralNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/LiteralNode.java @@ -37,7 +37,6 @@ import jdk.nashorn.internal.parser.Token; import jdk.nashorn.internal.parser.TokenType; import jdk.nashorn.internal.runtime.JSType; import jdk.nashorn.internal.runtime.ScriptRuntime; -import jdk.nashorn.internal.runtime.Source; import jdk.nashorn.internal.runtime.Undefined; /** @@ -50,16 +49,15 @@ public abstract class LiteralNode extends Node implements PropertyKey { /** Literal value */ protected final T value; - /** + /** * Constructor * - * @param source the source * @param token token * @param finish finish * @param value the value of the literal */ - protected LiteralNode(final Source source, final long token, final int finish, final T value) { - super(source, token, finish); + protected LiteralNode(final long token, final int finish, final T value) { + super(token, finish); this.value = value; } @@ -238,14 +236,13 @@ public abstract class LiteralNode extends Node implements PropertyKey { /** * Create a new null literal * - * @param source the source * @param token token * @param finish finish * * @return the new literal node */ - public static LiteralNode newInstance(final Source source, final long token, final int finish) { - return new NodeLiteralNode(source, token, finish); + public static LiteralNode newInstance(final long token, final int finish) { + return new NodeLiteralNode(token, finish); } /** @@ -256,14 +253,14 @@ public abstract class LiteralNode extends Node implements PropertyKey { * @return the new literal node */ public static LiteralNode newInstance(final Node parent) { - return new NodeLiteralNode(parent.getSource(), parent.getToken(), parent.getFinish()); + return new NodeLiteralNode(parent.getToken(), parent.getFinish()); } @Immutable private static final class BooleanLiteralNode extends LiteralNode { - private BooleanLiteralNode(final Source source, final long token, final int finish, final boolean value) { - super(source, Token.recast(token, value ? TokenType.TRUE : TokenType.FALSE), finish, value); + private BooleanLiteralNode(final long token, final int finish, final boolean value) { + super(Token.recast(token, value ? TokenType.TRUE : TokenType.FALSE), finish, value); } private BooleanLiteralNode(final BooleanLiteralNode literalNode) { @@ -289,15 +286,14 @@ public abstract class LiteralNode extends Node implements PropertyKey { /** * Create a new boolean literal * - * @param source the source * @param token token * @param finish finish * @param value true or false * * @return the new literal node */ - public static LiteralNode newInstance(final Source source, final long token, final int finish, final boolean value) { - return new BooleanLiteralNode(source, token, finish, value); + public static LiteralNode newInstance(final long token, final int finish, final boolean value) { + return new BooleanLiteralNode(token, finish, value); } /** @@ -309,7 +305,7 @@ public abstract class LiteralNode extends Node implements PropertyKey { * @return the new literal node */ public static LiteralNode newInstance(final Node parent, final boolean value) { - return new BooleanLiteralNode(parent.getSource(), parent.getToken(), parent.getFinish(), value); + return new BooleanLiteralNode(parent.getToken(), parent.getFinish(), value); } @Immutable @@ -317,8 +313,8 @@ public abstract class LiteralNode extends Node implements PropertyKey { private final Type type = numberGetType(value); - private NumberLiteralNode(final Source source, final long token, final int finish, final Number value) { - super(source, Token.recast(token, TokenType.DECIMAL), finish, value); + private NumberLiteralNode(final long token, final int finish, final Number value) { + super(Token.recast(token, TokenType.DECIMAL), finish, value); } private NumberLiteralNode(final NumberLiteralNode literalNode) { @@ -353,15 +349,14 @@ public abstract class LiteralNode extends Node implements PropertyKey { /** * Create a new number literal * - * @param source the source * @param token token * @param finish finish * @param value literal value * * @return the new literal node */ - public static LiteralNode newInstance(final Source source, final long token, final int finish, final Number value) { - return new NumberLiteralNode(source, token, finish, value); + public static LiteralNode newInstance(final long token, final int finish, final Number value) { + return new NumberLiteralNode(token, finish, value); } /** @@ -373,12 +368,12 @@ public abstract class LiteralNode extends Node implements PropertyKey { * @return the new literal node */ public static LiteralNode newInstance(final Node parent, final Number value) { - return new NumberLiteralNode(parent.getSource(), parent.getToken(), parent.getFinish(), value); + return new NumberLiteralNode(parent.getToken(), parent.getFinish(), value); } private static class UndefinedLiteralNode extends LiteralNode { - private UndefinedLiteralNode(final Source source, final long token, final int finish) { - super(source, Token.recast(token, TokenType.OBJECT), finish, ScriptRuntime.UNDEFINED); + private UndefinedLiteralNode(final long token, final int finish) { + super(Token.recast(token, TokenType.OBJECT), finish, ScriptRuntime.UNDEFINED); } private UndefinedLiteralNode(final UndefinedLiteralNode literalNode) { @@ -389,15 +384,14 @@ public abstract class LiteralNode extends Node implements PropertyKey { /** * Create a new undefined literal * - * @param source the source * @param token token * @param finish finish * @param value undefined value, passed only for polymorphisism discrimination * * @return the new literal node */ - public static LiteralNode newInstance(final Source source, final long token, final int finish, final Undefined value) { - return new UndefinedLiteralNode(source, token, finish); + public static LiteralNode newInstance(final long token, final int finish, final Undefined value) { + return new UndefinedLiteralNode(token, finish); } /** @@ -409,13 +403,13 @@ public abstract class LiteralNode extends Node implements PropertyKey { * @return the new literal node */ public static LiteralNode newInstance(final Node parent, final Undefined value) { - return new UndefinedLiteralNode(parent.getSource(), parent.getToken(), parent.getFinish()); + return new UndefinedLiteralNode(parent.getToken(), parent.getFinish()); } @Immutable private static class StringLiteralNode extends LiteralNode { - private StringLiteralNode(final Source source, final long token, final int finish, final String value) { - super(source, Token.recast(token, TokenType.STRING), finish, value); + private StringLiteralNode(final long token, final int finish, final String value) { + super(Token.recast(token, TokenType.STRING), finish, value); } private StringLiteralNode(final StringLiteralNode literalNode) { @@ -433,15 +427,14 @@ public abstract class LiteralNode extends Node implements PropertyKey { /** * Create a new string literal * - * @param source the source * @param token token * @param finish finish * @param value string value * * @return the new literal node */ - public static LiteralNode newInstance(final Source source, final long token, final int finish, final String value) { - return new StringLiteralNode(source, token, finish, value); + public static LiteralNode newInstance(final long token, final int finish, final String value) { + return new StringLiteralNode(token, finish, value); } /** @@ -453,13 +446,13 @@ public abstract class LiteralNode extends Node implements PropertyKey { * @return the new literal node */ public static LiteralNode newInstance(final Node parent, final String value) { - return new StringLiteralNode(parent.getSource(), parent.getToken(), parent.getFinish(), value); + return new StringLiteralNode(parent.getToken(), parent.getFinish(), value); } @Immutable private static class LexerTokenLiteralNode extends LiteralNode { - private LexerTokenLiteralNode(final Source source, final long token, final int finish, final LexerToken value) { - super(source, Token.recast(token, TokenType.STRING), finish, value); //TODO is string the correct token type here? + private LexerTokenLiteralNode(final long token, final int finish, final LexerToken value) { + super(Token.recast(token, TokenType.STRING), finish, value); //TODO is string the correct token type here? } private LexerTokenLiteralNode(final LexerTokenLiteralNode literalNode) { @@ -480,15 +473,14 @@ public abstract class LiteralNode extends Node implements PropertyKey { /** * Create a new literal node for a lexer token * - * @param source the source * @param token token * @param finish finish * @param value lexer token value * * @return the new literal node */ - public static LiteralNode newInstance(final Source source, final long token, final int finish, final LexerToken value) { - return new LexerTokenLiteralNode(source, token, finish, value); + public static LiteralNode newInstance(final long token, final int finish, final LexerToken value) { + return new LexerTokenLiteralNode(token, finish, value); } /** @@ -500,17 +492,17 @@ public abstract class LiteralNode extends Node implements PropertyKey { * @return the new literal node */ public static LiteralNode newInstance(final Node parent, final LexerToken value) { - return new LexerTokenLiteralNode(parent.getSource(), parent.getToken(), parent.getFinish(), value); + return new LexerTokenLiteralNode(parent.getToken(), parent.getFinish(), value); } private static final class NodeLiteralNode extends LiteralNode { - private NodeLiteralNode(final Source source, final long token, final int finish) { - this(source, token, finish, null); + private NodeLiteralNode(final long token, final int finish) { + this(token, finish, null); } - private NodeLiteralNode(final Source source, final long token, final int finish, final Node value) { - super(source, Token.recast(token, TokenType.OBJECT), finish, value); + private NodeLiteralNode(final long token, final int finish, final Node value) { + super(Token.recast(token, TokenType.OBJECT), finish, value); } private NodeLiteralNode(final LiteralNode literalNode) { @@ -550,15 +542,14 @@ public abstract class LiteralNode extends Node implements PropertyKey { /** * Create a new node literal for an arbitrary node * - * @param source the source * @param token token * @param finish finish * @param value the literal value node * * @return the new literal node */ - public static LiteralNode newInstance(final Source source, final long token, final int finish, final Node value) { - return new NodeLiteralNode(source, token, finish, value); + public static LiteralNode newInstance(final long token, final int finish, final Node value) { + return new NodeLiteralNode(token, finish, value); } /** @@ -570,7 +561,7 @@ public abstract class LiteralNode extends Node implements PropertyKey { * @return the new literal node */ public static LiteralNode newInstance(final Node parent, final Node value) { - return new NodeLiteralNode(parent.getSource(), parent.getToken(), parent.getFinish(), value); + return new NodeLiteralNode(parent.getToken(), parent.getFinish(), value); } /** @@ -645,13 +636,12 @@ public abstract class LiteralNode extends Node implements PropertyKey { /** * Constructor * - * @param source the source * @param token token * @param finish finish * @param value array literal value, a Node array */ - protected ArrayLiteralNode(final Source source, final long token, final int finish, final Node[] value) { - super(source, Token.recast(token, TokenType.ARRAY), finish, value); + protected ArrayLiteralNode(final long token, final int finish, final Node[] value) { + super(Token.recast(token, TokenType.ARRAY), finish, value); this.elementType = Type.UNKNOWN; } @@ -886,15 +876,14 @@ public abstract class LiteralNode extends Node implements PropertyKey { /** * Create a new array literal of Nodes from a list of Node values * - * @param source the source * @param token token * @param finish finish * @param value literal value list * * @return the new literal node */ - public static LiteralNode newInstance(final Source source, final long token, final int finish, final List value) { - return new ArrayLiteralNode(source, token, finish, value.toArray(new Node[value.size()])); + public static LiteralNode newInstance(final long token, final int finish, final List value) { + return new ArrayLiteralNode(token, finish, value.toArray(new Node[value.size()])); } @@ -907,20 +896,19 @@ public abstract class LiteralNode extends Node implements PropertyKey { * @return the new literal node */ public static LiteralNode newInstance(final Node parent, final List value) { - return new ArrayLiteralNode(parent.getSource(), parent.getToken(), parent.getFinish(), value.toArray(new Node[value.size()])); + return new ArrayLiteralNode(parent.getToken(), parent.getFinish(), value.toArray(new Node[value.size()])); } /** * Create a new array literal of Nodes * - * @param source the source * @param token token * @param finish finish * @param value literal value array * * @return the new literal node */ - public static LiteralNode newInstance(final Source source, final long token, final int finish, final Node[] value) { - return new ArrayLiteralNode(source, token, finish, value); + public static LiteralNode newInstance(final long token, final int finish, final Node[] value) { + return new ArrayLiteralNode(token, finish, value); } } diff --git a/nashorn/src/jdk/nashorn/internal/ir/Location.java b/nashorn/src/jdk/nashorn/internal/ir/Location.java deleted file mode 100644 index cd9edb5bd1d..00000000000 --- a/nashorn/src/jdk/nashorn/internal/ir/Location.java +++ /dev/null @@ -1,134 +0,0 @@ -/* - * Copyright (c) 2010, 2013, Oracle and/or its affiliates. All rights reserved. - * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. - * - * This code is free software; you can redistribute it and/or modify it - * under the terms of the GNU General Public License version 2 only, as - * published by the Free Software Foundation. Oracle designates this - * particular file as subject to the "Classpath" exception as provided - * by Oracle in the LICENSE file that accompanied this code. - * - * This code is distributed in the hope that it will be useful, but WITHOUT - * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or - * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License - * version 2 for more details (a copy is included in the LICENSE file that - * accompanied this code). - * - * You should have received a copy of the GNU General Public License version - * 2 along with this work; if not, write to the Free Software Foundation, - * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. - * - * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA - * or visit www.oracle.com if you need additional information or have any - * questions. - */ - -package jdk.nashorn.internal.ir; - -import jdk.nashorn.internal.parser.Token; -import jdk.nashorn.internal.parser.TokenType; -import jdk.nashorn.internal.runtime.Source; - -/** - * Used to locate an entity back to it's source file. - */ -public class Location implements Cloneable { - /** Source of entity. */ - private final Source source; - - /** Token descriptor. */ - private final long token; - - /** - * Constructor - * - * @param source the source - * @param token token - */ - public Location(final Source source, final long token) { - this.source = source; - this.token = token; - } - - /** - * Copy constructor - * - * @param location source node - */ - protected Location(final Location location) { - this.source = location.source; - this.token = location.token; - } - - @Override - protected Object clone() { - try { - return super.clone(); - } catch(CloneNotSupportedException e) { - throw new AssertionError(e); - } - } - - @Override - public final boolean equals(final Object other) { - return super.equals(other); - } - - @Override - public final int hashCode() { - return super.hashCode(); - } - - /** - * Return token position from a token descriptor. - * - * @return Start position of the token in the source. - */ - public int position() { - return Token.descPosition(token); - } - - /** - * Return token length from a token descriptor. - * - * @return Length of the token. - */ - public int length() { - return Token.descLength(token); - } - - /** - * Return token tokenType from a token descriptor. - * - * @return Type of token. - */ - public TokenType tokenType() { - return Token.descType(token); - } - - /** - * Test token tokenType. - * - * @param type a type to check this token against - * @return true if token types match. - */ - public boolean isTokenType(final TokenType type) { - return Token.descType(token) == type; - } - - /** - * Get the source for this location - * @return the source - */ - public Source getSource() { - return source; - } - - /** - * Get the token for this location - * @return the token - */ - public long getToken() { - return token; - } -} diff --git a/nashorn/src/jdk/nashorn/internal/ir/LoopNode.java b/nashorn/src/jdk/nashorn/internal/ir/LoopNode.java index b3909dc266d..f4bbebe4551 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/LoopNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/LoopNode.java @@ -29,7 +29,6 @@ import java.util.Arrays; import java.util.List; import jdk.nashorn.internal.codegen.Label; -import jdk.nashorn.internal.runtime.Source; /** * A loop node, for example a while node, do while node or for node @@ -50,15 +49,14 @@ public abstract class LoopNode extends BreakableNode { /** * Constructor * - * @param source source * @param token token * @param finish finish * @param test test, or null if infinite loop * @param body loop body * @param controlFlowEscapes controlFlowEscapes */ - protected LoopNode(final Source source, final long token, final int finish, final Node test, final Block body, final boolean controlFlowEscapes) { - super(source, token, finish, new Label("while_break")); + protected LoopNode(final long token, final int finish, final Node test, final Block body, final boolean controlFlowEscapes) { + super(token, finish, new Label("while_break")); this.continueLabel = new Label("while_continue"); this.test = test; this.body = body; diff --git a/nashorn/src/jdk/nashorn/internal/ir/Node.java b/nashorn/src/jdk/nashorn/internal/ir/Node.java index d370f22f12f..a86c57f4daf 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/Node.java +++ b/nashorn/src/jdk/nashorn/internal/ir/Node.java @@ -31,12 +31,12 @@ import java.util.List; import jdk.nashorn.internal.codegen.types.Type; import jdk.nashorn.internal.ir.visitor.NodeVisitor; import jdk.nashorn.internal.parser.Token; -import jdk.nashorn.internal.runtime.Source; +import jdk.nashorn.internal.parser.TokenType; /** * Nodes are used to compose Abstract Syntax Trees. */ -public abstract class Node extends Location { +public abstract class Node implements Cloneable { /** Node symbol. */ private Symbol symbol; @@ -46,16 +46,17 @@ public abstract class Node extends Location { /** End of source range. */ protected int finish; + /** Token descriptor. */ + private final long token; + /** * Constructor * - * @param source the source * @param token token * @param finish finish */ - public Node(final Source source, final long token, final int finish) { - super(source, token); - + public Node(final long token, final int finish) { + this.token = token; this.start = Token.descPosition(token); this.finish = finish; } @@ -63,16 +64,14 @@ public abstract class Node extends Location { /** * Constructor * - * @param source source * @param token token * @param start start * @param finish finish */ - protected Node(final Source source, final long token, final int start, final int finish) { - super(source, token); - + protected Node(final long token, final int start, final int finish) { this.start = start; this.finish = finish; + this.token = token; } /** @@ -81,8 +80,7 @@ public abstract class Node extends Location { * @param node source node */ protected Node(final Node node) { - super(node); - + this.token = node.token; this.symbol = node.symbol; this.start = node.start; this.finish = node.finish; @@ -248,6 +246,15 @@ public abstract class Node extends Location { return symbol; } + @Override + protected Object clone() { + try { + return super.clone(); + } catch (final CloneNotSupportedException e) { + throw new AssertionError(e); + } + } + /** * Assign a symbol to this node. See {@link Node#getSymbol()} for explanation * of what a symbol is @@ -265,6 +272,62 @@ public abstract class Node extends Location { return newNode; } + + @Override + public final boolean equals(final Object other) { + return super.equals(other); + } + + @Override + public final int hashCode() { + return super.hashCode(); + } + + /** + * Return token position from a token descriptor. + * + * @return Start position of the token in the source. + */ + public int position() { + return Token.descPosition(token); + } + + /** + * Return token length from a token descriptor. + * + * @return Length of the token. + */ + public int length() { + return Token.descLength(token); + } + + /** + * Return token tokenType from a token descriptor. + * + * @return Type of token. + */ + public TokenType tokenType() { + return Token.descType(token); + } + + /** + * Test token tokenType. + * + * @param type a type to check this token against + * @return true if token types match. + */ + public boolean isTokenType(final TokenType type) { + return Token.descType(token) == type; + } + + /** + * Get the token for this location + * @return the token + */ + public long getToken() { + return token; + } + /** * Is this a terminal Node, i.e. does it end control flow like a throw or return * expression does? diff --git a/nashorn/src/jdk/nashorn/internal/ir/ObjectNode.java b/nashorn/src/jdk/nashorn/internal/ir/ObjectNode.java index 744a44d7a56..34069725640 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/ObjectNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/ObjectNode.java @@ -30,7 +30,6 @@ import java.util.List; import jdk.nashorn.internal.ir.annotations.Immutable; import jdk.nashorn.internal.ir.visitor.NodeVisitor; -import jdk.nashorn.internal.runtime.Source; /** * IR representation of an object literal. @@ -44,13 +43,12 @@ public final class ObjectNode extends Node { /** * Constructor * - * @param source the source * @param token token * @param finish finish * @param elements the elements used to initialize this ObjectNode */ - public ObjectNode(final Source source, final long token, final int finish, final List elements) { - super(source, token, finish); + public ObjectNode(final long token, final int finish, final List elements) { + super(token, finish); this.elements = elements; } diff --git a/nashorn/src/jdk/nashorn/internal/ir/PropertyNode.java b/nashorn/src/jdk/nashorn/internal/ir/PropertyNode.java index 635e1aa3889..7040d41477d 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/PropertyNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/PropertyNode.java @@ -27,7 +27,6 @@ package jdk.nashorn.internal.ir; import jdk.nashorn.internal.ir.annotations.Immutable; import jdk.nashorn.internal.ir.visitor.NodeVisitor; -import jdk.nashorn.internal.runtime.Source; /** * IR representation of an object literal property. @@ -50,7 +49,6 @@ public final class PropertyNode extends Node { /** * Constructor * - * @param source the source * @param token token * @param finish finish * @param key the key of this property @@ -58,8 +56,8 @@ public final class PropertyNode extends Node { * @param getter getter function body * @param setter setter function body */ - public PropertyNode(final Source source, final long token, final int finish, final PropertyKey key, final Node value, final FunctionNode getter, final FunctionNode setter) { - super(source, token, finish); + public PropertyNode(final long token, final int finish, final PropertyKey key, final Node value, final FunctionNode getter, final FunctionNode setter) { + super(token, finish); this.key = key; this.value = value; this.getter = getter; diff --git a/nashorn/src/jdk/nashorn/internal/ir/ReturnNode.java b/nashorn/src/jdk/nashorn/internal/ir/ReturnNode.java index dafc956deec..cb736e427ed 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/ReturnNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/ReturnNode.java @@ -29,7 +29,6 @@ import static jdk.nashorn.internal.parser.TokenType.RETURN; import static jdk.nashorn.internal.parser.TokenType.YIELD; import jdk.nashorn.internal.ir.annotations.Immutable; import jdk.nashorn.internal.ir.visitor.NodeVisitor; -import jdk.nashorn.internal.runtime.Source; /** * IR representation for RETURN or YIELD statements. @@ -42,13 +41,12 @@ public class ReturnNode extends Node { /** * Constructor * - * @param source the source * @param token token * @param finish finish * @param expression expression to return */ - public ReturnNode(final Source source, final long token, final int finish, final Node expression) { - super(source, token, finish); + public ReturnNode(final long token, final int finish, final Node expression) { + super(token, finish); this.expression = expression; } diff --git a/nashorn/src/jdk/nashorn/internal/ir/RuntimeNode.java b/nashorn/src/jdk/nashorn/internal/ir/RuntimeNode.java index 7bdb6c0adb8..ceef9b9b823 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/RuntimeNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/RuntimeNode.java @@ -33,7 +33,6 @@ import jdk.nashorn.internal.codegen.types.Type; import jdk.nashorn.internal.ir.annotations.Immutable; import jdk.nashorn.internal.ir.visitor.NodeVisitor; import jdk.nashorn.internal.parser.TokenType; -import jdk.nashorn.internal.runtime.Source; /** * IR representation for a runtime call. @@ -280,14 +279,13 @@ public class RuntimeNode extends Node implements TypeOverride { /** * Constructor * - * @param source the source * @param token token * @param finish finish * @param request the request * @param args arguments to request */ - public RuntimeNode(final Source source, final long token, final int finish, final Request request, final List args) { - super(source, token, finish); + public RuntimeNode(final long token, final int finish, final Request request, final List args) { + super(token, finish); this.request = request; this.args = args; @@ -307,14 +305,13 @@ public class RuntimeNode extends Node implements TypeOverride { /** * Constructor * - * @param source the source * @param token token * @param finish finish * @param request the request * @param args arguments to request */ - public RuntimeNode(final Source source, final long token, final int finish, final Request request, final Node... args) { - this(source, token, finish, request, Arrays.asList(args)); + public RuntimeNode(final long token, final int finish, final Request request, final Node... args) { + this(token, finish, request, Arrays.asList(args)); } /** diff --git a/nashorn/src/jdk/nashorn/internal/ir/SplitNode.java b/nashorn/src/jdk/nashorn/internal/ir/SplitNode.java index 49c4092f221..05b2b04b20d 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/SplitNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/SplitNode.java @@ -51,7 +51,7 @@ public class SplitNode extends LexicalContextNode { * @param compileUnit compile unit to use for the body */ public SplitNode(final String name, final Node body, final CompileUnit compileUnit) { - super(body.getSource(), body.getToken(), body.getFinish()); + super(body.getToken(), body.getFinish()); this.name = name; this.body = body; this.compileUnit = compileUnit; diff --git a/nashorn/src/jdk/nashorn/internal/ir/SwitchNode.java b/nashorn/src/jdk/nashorn/internal/ir/SwitchNode.java index 7864a10d0a8..640911148c6 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/SwitchNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/SwitchNode.java @@ -32,7 +32,6 @@ import java.util.List; import jdk.nashorn.internal.codegen.Label; import jdk.nashorn.internal.ir.annotations.Immutable; import jdk.nashorn.internal.ir.visitor.NodeVisitor; -import jdk.nashorn.internal.runtime.Source; /** * IR representation of a SWITCH statement. @@ -54,15 +53,14 @@ public final class SwitchNode extends BreakableNode { /** * Constructor * - * @param source the source * @param token token * @param finish finish * @param expression switch expression * @param cases cases * @param defaultCase the default case node - null if none, otherwise has to be present in cases list */ - public SwitchNode(final Source source, final long token, final int finish, final Node expression, final List cases, final CaseNode defaultCase) { - super(source, token, finish, new Label("switch_break")); + public SwitchNode(final long token, final int finish, final Node expression, final List cases, final CaseNode defaultCase) { + super(token, finish, new Label("switch_break")); this.expression = expression; this.cases = cases; this.defaultCaseIndex = defaultCase == null ? -1 : cases.indexOf(defaultCase); diff --git a/nashorn/src/jdk/nashorn/internal/ir/TernaryNode.java b/nashorn/src/jdk/nashorn/internal/ir/TernaryNode.java index e2ccdb91173..25ac1a4ddaa 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/TernaryNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/TernaryNode.java @@ -27,7 +27,6 @@ package jdk.nashorn.internal.ir; import jdk.nashorn.internal.ir.annotations.Immutable; import jdk.nashorn.internal.ir.visitor.NodeVisitor; -import jdk.nashorn.internal.runtime.Source; /** * TernaryNode nodes represent three operand operations (?:). @@ -44,14 +43,13 @@ public final class TernaryNode extends Node { /** * Constructor * - * @param source the source * @param token token * @param lhs left hand side node * @param rhs right hand side node * @param third third node */ - public TernaryNode(final Source source, final long token, final Node lhs, final Node rhs, final Node third) { - super(source, token, third.getFinish()); + public TernaryNode(final long token, final Node lhs, final Node rhs, final Node third) { + super(token, third.getFinish()); this.lhs = lhs; this.rhs = rhs; this.third = third; diff --git a/nashorn/src/jdk/nashorn/internal/ir/ThrowNode.java b/nashorn/src/jdk/nashorn/internal/ir/ThrowNode.java index 7a10a6add83..d7e6575b160 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/ThrowNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/ThrowNode.java @@ -27,7 +27,6 @@ package jdk.nashorn.internal.ir; import jdk.nashorn.internal.ir.annotations.Immutable; import jdk.nashorn.internal.ir.visitor.NodeVisitor; -import jdk.nashorn.internal.runtime.Source; /** * IR representation for THROW statements. @@ -40,14 +39,12 @@ public final class ThrowNode extends Node { /** * Constructor * - * @param source the source * @param token token * @param finish finish * @param expression expression to throw */ - public ThrowNode(final Source source, final long token, final int finish, final Node expression) { - super(source, token, finish); - + public ThrowNode(final long token, final int finish, final Node expression) { + super(token, finish); this.expression = expression; } diff --git a/nashorn/src/jdk/nashorn/internal/ir/TryNode.java b/nashorn/src/jdk/nashorn/internal/ir/TryNode.java index 5e3ff7af135..0cca5ea86b0 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/TryNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/TryNode.java @@ -32,7 +32,6 @@ import java.util.List; import jdk.nashorn.internal.codegen.Label; import jdk.nashorn.internal.ir.annotations.Immutable; import jdk.nashorn.internal.ir.visitor.NodeVisitor; -import jdk.nashorn.internal.runtime.Source; /** * IR representation of a TRY statement. @@ -60,15 +59,14 @@ public final class TryNode extends Node { /** * Constructor * - * @param source the source * @param token token * @param finish finish * @param body try node body * @param catchBlocks list of catch blocks in order * @param finallyBody body of finally block or null if none */ - public TryNode(final Source source, final long token, final int finish, final Block body, final List catchBlocks, final Block finallyBody) { - super(source, token, finish); + public TryNode(final long token, final int finish, final Block body, final List catchBlocks, final Block finallyBody) { + super(token, finish); this.body = body; this.catchBlocks = catchBlocks; this.finallyBody = finallyBody; diff --git a/nashorn/src/jdk/nashorn/internal/ir/UnaryNode.java b/nashorn/src/jdk/nashorn/internal/ir/UnaryNode.java index fed5e408293..ed1d8a9bdbe 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/UnaryNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/UnaryNode.java @@ -35,7 +35,6 @@ import jdk.nashorn.internal.ir.annotations.Immutable; import jdk.nashorn.internal.ir.visitor.NodeVisitor; import jdk.nashorn.internal.parser.Token; import jdk.nashorn.internal.parser.TokenType; -import jdk.nashorn.internal.runtime.Source; /** * UnaryNode nodes represent single operand operations. @@ -48,24 +47,23 @@ public final class UnaryNode extends Node implements Assignment { /** * Constructor * - * @param source the source * @param token token * @param rhs expression */ - public UnaryNode(final Source source, final long token, final Node rhs) { - this(source, token, Math.min(rhs.getStart(), Token.descPosition(token)), Math.max(Token.descPosition(token) + Token.descLength(token), rhs.getFinish()), rhs); + public UnaryNode(final long token, final Node rhs) { + this(token, Math.min(rhs.getStart(), Token.descPosition(token)), Math.max(Token.descPosition(token) + Token.descLength(token), rhs.getFinish()), rhs); } /** * Constructor - * @param source the source + * * @param token token * @param start start * @param finish finish * @param rhs expression */ - public UnaryNode(final Source source, final long token, final int start, final int finish, final Node rhs) { - super(source, token, start, finish); + public UnaryNode(final long token, final int start, final int finish, final Node rhs) { + super(token, start, finish); this.rhs = rhs; } diff --git a/nashorn/src/jdk/nashorn/internal/ir/VarNode.java b/nashorn/src/jdk/nashorn/internal/ir/VarNode.java index fbc3eabecd2..ae3518a767f 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/VarNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/VarNode.java @@ -27,7 +27,6 @@ package jdk.nashorn.internal.ir; import jdk.nashorn.internal.ir.annotations.Immutable; import jdk.nashorn.internal.ir.visitor.NodeVisitor; -import jdk.nashorn.internal.runtime.Source; /** * Node represents a var/let declaration. @@ -54,14 +53,13 @@ public final class VarNode extends Node implements Assignment { /** * Constructor * - * @param source the source * @param token token * @param finish finish * @param name name of variable * @param init init node or null if just a declaration */ - public VarNode(final Source source, final long token, final int finish, final IdentNode name, final Node init) { - this(source, token, finish, name, init, IS_STATEMENT); + public VarNode(final long token, final int finish, final IdentNode name, final Node init) { + this(token, finish, name, init, IS_STATEMENT); } private VarNode(final VarNode varNode, final IdentNode name, final Node init, final int flags) { @@ -74,15 +72,14 @@ public final class VarNode extends Node implements Assignment { /** * Constructor * - * @param source the source * @param token token * @param finish finish * @param name name of variable * @param init init node or null if just a declaration * @param flags flags */ - public VarNode(final Source source, final long token, final int finish, final IdentNode name, final Node init, final int flags) { - super(source, token, finish); + public VarNode(final long token, final int finish, final IdentNode name, final Node init, final int flags) { + super(token, finish); this.name = init == null ? name : name.setIsInitializedHere(); this.init = init; diff --git a/nashorn/src/jdk/nashorn/internal/ir/WhileNode.java b/nashorn/src/jdk/nashorn/internal/ir/WhileNode.java index 438be0131e6..fd347947b41 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/WhileNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/WhileNode.java @@ -27,7 +27,6 @@ package jdk.nashorn.internal.ir; import jdk.nashorn.internal.ir.annotations.Immutable; import jdk.nashorn.internal.ir.visitor.NodeVisitor; -import jdk.nashorn.internal.runtime.Source; /** * IR representation for a WHILE statement. This is the superclass of all @@ -42,13 +41,12 @@ public final class WhileNode extends LoopNode { /** * Constructor * - * @param source the source * @param token token * @param finish finish * @param isDoWhile is this a do while loop? */ - public WhileNode(final Source source, final long token, final int finish, final boolean isDoWhile) { - super(source, token, finish, null, null, false); + public WhileNode(final long token, final int finish, final boolean isDoWhile) { + super(token, finish, null, null, false); this.isDoWhile = isDoWhile; } diff --git a/nashorn/src/jdk/nashorn/internal/ir/WithNode.java b/nashorn/src/jdk/nashorn/internal/ir/WithNode.java index 5ebbfd55d6d..64d735b5eec 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/WithNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/WithNode.java @@ -27,7 +27,6 @@ package jdk.nashorn.internal.ir; import jdk.nashorn.internal.ir.annotations.Immutable; import jdk.nashorn.internal.ir.visitor.NodeVisitor; -import jdk.nashorn.internal.runtime.Source; /** * IR representation for {@code with} statements. @@ -43,20 +42,17 @@ public final class WithNode extends LexicalContextNode { /** * Constructor * - * @param source the source * @param token token * @param finish finish */ - public WithNode(final Source source, final long token, final int finish) { - super(source, token, finish); - + public WithNode(final long token, final int finish) { + super(token, finish); this.expression = null; this.body = null; } private WithNode(final WithNode node, final Node expression, final Block body) { super(node); - this.expression = expression; this.body = body; } diff --git a/nashorn/src/jdk/nashorn/internal/ir/debug/JSONWriter.java b/nashorn/src/jdk/nashorn/internal/ir/debug/JSONWriter.java index 988b756c9fd..24273ac1304 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/debug/JSONWriter.java +++ b/nashorn/src/jdk/nashorn/internal/ir/debug/JSONWriter.java @@ -971,7 +971,7 @@ public final class JSONWriter extends NodeVisitor { objectStart("loc"); // source name - final Source src = node.getSource(); + final Source src = getLexicalContext().getCurrentFunction().getSource(); property("source", src.getName()); comma(); diff --git a/nashorn/src/jdk/nashorn/internal/objects/NativeDebug.java b/nashorn/src/jdk/nashorn/internal/objects/NativeDebug.java index 337f2e3d5b1..bcc9b5c2318 100644 --- a/nashorn/src/jdk/nashorn/internal/objects/NativeDebug.java +++ b/nashorn/src/jdk/nashorn/internal/objects/NativeDebug.java @@ -178,7 +178,6 @@ public final class NativeDebug extends ScriptObject { * @param self self reference * @return undefined */ - @SuppressWarnings("resource") @Function(attributes = Attribute.NOT_ENUMERABLE, where = Where.CONSTRUCTOR) public static Object dumpCounters(final Object self) { final PrintWriter out = Context.getCurrentErr(); diff --git a/nashorn/src/jdk/nashorn/internal/parser/AbstractParser.java b/nashorn/src/jdk/nashorn/internal/parser/AbstractParser.java index b87d9cf3fea..25b500cef07 100644 --- a/nashorn/src/jdk/nashorn/internal/parser/AbstractParser.java +++ b/nashorn/src/jdk/nashorn/internal/parser/AbstractParser.java @@ -364,7 +364,7 @@ public abstract class AbstractParser { next(); // Create IDENT node. - return new IdentNode(source, identToken, finish, ident); + return new IdentNode(identToken, finish, ident); } // Get IDENT. @@ -373,7 +373,7 @@ public abstract class AbstractParser { return null; } // Create IDENT node. - return new IdentNode(source, identToken, finish, ident); + return new IdentNode(identToken, finish, ident); } /** @@ -408,7 +408,7 @@ public abstract class AbstractParser { final String ident = (String)getValue(identToken); next(); // Create IDENT node. - return new IdentNode(source, identToken, finish, ident); + return new IdentNode(identToken, finish, ident); } else { expect(IDENT); return null; @@ -433,11 +433,11 @@ public abstract class AbstractParser { LiteralNode node = null; if (value == null) { - node = LiteralNode.newInstance(source, literalToken, finish); + node = LiteralNode.newInstance(literalToken, finish); } else if (value instanceof Number) { - node = LiteralNode.newInstance(source, literalToken, finish, (Number)value); + node = LiteralNode.newInstance(literalToken, finish, (Number)value); } else if (value instanceof String) { - node = LiteralNode.newInstance(source, literalToken, finish, (String)value); + node = LiteralNode.newInstance(literalToken, finish, (String)value); } else if (value instanceof LexerToken) { if (value instanceof RegexToken) { final RegexToken regex = (RegexToken)value; @@ -447,7 +447,7 @@ public abstract class AbstractParser { throw error(e.getMessage()); } } - node = LiteralNode.newInstance(source, literalToken, finish, (LexerToken)value); + node = LiteralNode.newInstance(literalToken, finish, (LexerToken)value); } else { assert false : "unknown type for LiteralNode: " + value.getClass(); } diff --git a/nashorn/src/jdk/nashorn/internal/parser/JSONParser.java b/nashorn/src/jdk/nashorn/internal/parser/JSONParser.java index e074cf01a1c..d4985ea337e 100644 --- a/nashorn/src/jdk/nashorn/internal/parser/JSONParser.java +++ b/nashorn/src/jdk/nashorn/internal/parser/JSONParser.java @@ -193,13 +193,13 @@ public class JSONParser extends AbstractParser { return getLiteral(); case FALSE: next(); - return LiteralNode.newInstance(source, literalToken, finish, false); + return LiteralNode.newInstance(literalToken, finish, false); case TRUE: next(); - return LiteralNode.newInstance(source, literalToken, finish, true); + return LiteralNode.newInstance(literalToken, finish, true); case NULL: next(); - return LiteralNode.newInstance(source, literalToken, finish); + return LiteralNode.newInstance(literalToken, finish); case LBRACKET: return arrayLiteral(); case LBRACE: @@ -218,7 +218,7 @@ public class JSONParser extends AbstractParser { if (value instanceof Number) { next(); - return new UnaryNode(source, literalToken, LiteralNode.newInstance(source, realToken, finish, (Number)value)); + return new UnaryNode(literalToken, LiteralNode.newInstance(realToken, finish, (Number)value)); } throw error(AbstractParser.message("expected", "number", type.getNameOrType())); @@ -250,7 +250,7 @@ loop: switch (type) { case RBRACKET: next(); - result = LiteralNode.newInstance(source, arrayToken, finish, elements); + result = LiteralNode.newInstance(arrayToken, finish, elements); break loop; case COMMARIGHT: @@ -310,7 +310,7 @@ loop: } // Construct new object literal. - return new ObjectNode(source, objectToken, finish, elements); + return new ObjectNode(objectToken, finish, elements); } /** @@ -331,7 +331,7 @@ loop: if (name != null) { expect(COLON); final Node value = jsonLiteral(); - return new PropertyNode(source, propertyToken, value.getFinish(), name, value, null, null); + return new PropertyNode(propertyToken, value.getFinish(), name, value, null, null); } // Raise an error. diff --git a/nashorn/src/jdk/nashorn/internal/parser/Parser.java b/nashorn/src/jdk/nashorn/internal/parser/Parser.java index 6302d1f7b03..05807bab2e5 100644 --- a/nashorn/src/jdk/nashorn/internal/parser/Parser.java +++ b/nashorn/src/jdk/nashorn/internal/parser/Parser.java @@ -275,8 +275,7 @@ loop: * @return New block. */ private Block newBlock() { - final Block block = new Block(source, token, Token.descPosition(token)); - return lc.push(block); + return lc.push(new Block(token, Token.descPosition(token))); } /** @@ -479,7 +478,7 @@ loop: } // Build up node. - return new BinaryNode(source, op, lhs, rhs); + return new BinaryNode(op, lhs, rhs); } /** @@ -490,12 +489,12 @@ loop: * @param isPostfix Prefix or postfix. * @return Reduced expression. */ - private Node incDecExpression(final long firstToken, final TokenType tokenType, final Node expression, final boolean isPostfix) { + private static Node incDecExpression(final long firstToken, final TokenType tokenType, final Node expression, final boolean isPostfix) { if (isPostfix) { - return new UnaryNode(source, Token.recast(firstToken, tokenType == DECPREFIX ? DECPOSTFIX : INCPOSTFIX), expression.getStart(), Token.descPosition(firstToken) + Token.descLength(firstToken), expression); + return new UnaryNode(Token.recast(firstToken, tokenType == DECPREFIX ? DECPOSTFIX : INCPOSTFIX), expression.getStart(), Token.descPosition(firstToken) + Token.descLength(firstToken), expression); } - return new UnaryNode(source, firstToken, expression); + return new UnaryNode(firstToken, expression); } /** @@ -524,7 +523,7 @@ loop: FunctionNode script = newFunctionNode( functionToken, - new IdentNode(source, functionToken, Token.descPosition(functionToken), scriptName), + new IdentNode(functionToken, Token.descPosition(functionToken), scriptName), new ArrayList(), FunctionNode.Kind.SCRIPT); @@ -774,7 +773,7 @@ loop: private void block() { final Block newBlock = getBlock(true); // Force block execution. - appendStatement(new ExecuteNode(source, newBlock.getToken(), finish, newBlock)); + appendStatement(new ExecuteNode(newBlock.getToken(), finish, newBlock)); } /** @@ -867,7 +866,7 @@ loop: } // Allocate var node. - final VarNode var = new VarNode(source, varToken, finish, name, init); + final VarNode var = new VarNode(varToken, finish, name, init); vars.add(var); appendStatement(var); @@ -899,7 +898,7 @@ loop: */ private void emptyStatement() { if (env._empty_statements) { - appendStatement(new EmptyNode(source, token, Token.descPosition(token) + Token.descLength(token))); + appendStatement(new EmptyNode(token, Token.descPosition(token) + Token.descLength(token))); } // SEMICOLON checked in caller. @@ -923,7 +922,7 @@ loop: ExecuteNode executeNode = null; if (expression != null) { - executeNode = new ExecuteNode(source, expressionToken, finish, expression); + executeNode = new ExecuteNode(expressionToken, finish, expression); appendStatement(executeNode); } else { expect(null); @@ -963,7 +962,7 @@ loop: fail = getStatement(); } - appendStatement(new IfNode(source, ifToken, fail != null ? fail.getFinish() : pass.getFinish(), test, pass, fail)); + appendStatement(new IfNode(ifToken, fail != null ? fail.getFinish() : pass.getFinish(), test, pass, fail)); } /** @@ -980,7 +979,7 @@ loop: */ private void forStatement() { // Create FOR node, capturing FOR token. - ForNode forNode = new ForNode(source, token, Token.descPosition(token), null, null, null, null, ForNode.IS_FOR); + ForNode forNode = new ForNode(token, Token.descPosition(token), null, null, null, null, ForNode.IS_FOR); // Set up new block for scope of vars. Captures first token. @@ -1084,7 +1083,7 @@ loop: outer = restoreBlock(outer); } - appendStatement(new ExecuteNode(source, outer.getToken(), outer.getFinish(), outer)); + appendStatement(new ExecuteNode(outer.getToken(), outer.getFinish(), outer)); } /** @@ -1120,7 +1119,7 @@ loop: next(); // Construct WHILE node. - WhileNode whileNode = new WhileNode(source, whileToken, Token.descPosition(whileToken), false); + WhileNode whileNode = new WhileNode(whileToken, Token.descPosition(whileToken), false); lc.push(whileNode); try { @@ -1150,7 +1149,7 @@ loop: // DO tested in the caller. next(); - WhileNode doWhileNode = new WhileNode(source, doToken, Token.descPosition(doToken), true); + WhileNode doWhileNode = new WhileNode(doToken, Token.descPosition(doToken), true); lc.push(doWhileNode); try { @@ -1216,7 +1215,7 @@ loop: endOfLine(); // Construct and add CONTINUE node. - appendStatement(new ContinueNode(source, continueToken, finish, label == null ? null : new IdentNode(label))); + appendStatement(new ContinueNode(continueToken, finish, label == null ? null : new IdentNode(label))); } /** @@ -1263,7 +1262,7 @@ loop: endOfLine(); // Construct and add BREAK node. - appendStatement(new BreakNode(source, breakToken, finish, label == null ? null : new IdentNode(label))); + appendStatement(new BreakNode(breakToken, finish, label == null ? null : new IdentNode(label))); } /** @@ -1302,7 +1301,7 @@ loop: endOfLine(); // Construct and add RETURN node. - appendStatement(new ReturnNode(source, returnToken, finish, expression)); + appendStatement(new ReturnNode(returnToken, finish, expression)); } /** @@ -1336,7 +1335,7 @@ loop: endOfLine(); // Construct and add YIELD node. - appendStatement(new ReturnNode(source, yieldToken, finish, expression)); + appendStatement(new ReturnNode(yieldToken, finish, expression)); } /** @@ -1359,7 +1358,7 @@ loop: } // Get WITH expression. - WithNode withNode = new WithNode(source, withToken, finish); + WithNode withNode = new WithNode(withToken, finish); try { lc.push(withNode); @@ -1402,7 +1401,7 @@ loop: next(); // Create and add switch statement. - SwitchNode switchNode = new SwitchNode(source, switchToken, Token.descPosition(switchToken), null, new ArrayList(), null); + SwitchNode switchNode = new SwitchNode(switchToken, Token.descPosition(switchToken), null, new ArrayList(), null); lc.push(switchNode); try { @@ -1444,7 +1443,7 @@ loop: // Get CASE body. final Block statements = getBlock(false); - final CaseNode caseNode = new CaseNode(source, caseToken, finish, caseExpression, statements); + final CaseNode caseNode = new CaseNode(caseToken, finish, caseExpression, statements); statements.setFinish(finish); if (caseExpression == null) { @@ -1484,7 +1483,7 @@ loop: throw error(AbstractParser.message("duplicate.label", ident.getName()), labelToken); } - LabelNode labelNode = new LabelNode(source, labelToken, finish, ident, null); + LabelNode labelNode = new LabelNode(labelToken, finish, ident, null); try { lc.push(labelNode); labelNode = labelNode.setBody(lc, getStatement()); @@ -1530,7 +1529,7 @@ loop: endOfLine(); - appendStatement(new ThrowNode(source, throwToken, finish, expression)); + appendStatement(new ThrowNode(throwToken, finish, expression)); } /** @@ -1588,7 +1587,7 @@ loop: try { // Get CATCH body. final Block catchBody = getBlock(true); - final CatchNode catchNode = new CatchNode(source, catchToken, finish, exception, ifExpression, catchBody); + final CatchNode catchNode = new CatchNode(catchToken, finish, exception, ifExpression, catchBody); appendStatement(catchNode); } finally { catchBlock = restoreBlock(catchBlock); @@ -1614,7 +1613,7 @@ loop: throw error(AbstractParser.message("missing.catch.or.finally"), tryToken); } - final TryNode tryNode = new TryNode(source, tryToken, Token.descPosition(tryToken), tryBody, catchBlocks, finallyStatements); + final TryNode tryNode = new TryNode(tryToken, Token.descPosition(tryToken), tryBody, catchBlocks, finallyStatements); // Add try. assert lc.peek() == outer; appendStatement(tryNode); @@ -1626,7 +1625,7 @@ loop: outer = restoreBlock(outer); } - appendStatement(new ExecuteNode(source, outer.getToken(), outer.getFinish(), outer)); + appendStatement(new ExecuteNode(outer.getToken(), outer.getFinish(), outer)); } /** @@ -1643,7 +1642,7 @@ loop: // DEBUGGER tested in caller. next(); endOfLine(); - appendStatement(new RuntimeNode(source, debuggerToken, finish, RuntimeNode.Request.DEBUGGER, new ArrayList())); + appendStatement(new RuntimeNode(debuggerToken, finish, RuntimeNode.Request.DEBUGGER, new ArrayList())); } /** @@ -1669,7 +1668,7 @@ loop: case THIS: final String name = type.getName(); next(); - return new IdentNode(source, primaryToken, finish, name); + return new IdentNode(primaryToken, finish, name); case IDENT: final IdentNode ident = getIdent(); if (ident == null) { @@ -1693,13 +1692,13 @@ loop: return execString(primaryToken); case FALSE: next(); - return LiteralNode.newInstance(source, primaryToken, finish, false); + return LiteralNode.newInstance(primaryToken, finish, false); case TRUE: next(); - return LiteralNode.newInstance(source, primaryToken, finish, true); + return LiteralNode.newInstance(primaryToken, finish, true); case NULL: next(); - return LiteralNode.newInstance(source, primaryToken, finish); + return LiteralNode.newInstance(primaryToken, finish); case LBRACKET: return arrayLiteral(); case LBRACE: @@ -1736,7 +1735,7 @@ loop: */ Node execString(final long primaryToken) { // Synthesize an ident to call $EXEC. - final IdentNode execIdent = new IdentNode(source, primaryToken, finish, ScriptingFunctions.EXEC_NAME); + final IdentNode execIdent = new IdentNode(primaryToken, finish, ScriptingFunctions.EXEC_NAME); // Skip over EXECSTRING. next(); // Set up argument list for call. @@ -1748,7 +1747,7 @@ loop: // Skip ending of edit string expression. expect(RBRACE); - return new CallNode(source, primaryToken, finish, execIdent, arguments); + return new CallNode(primaryToken, finish, execIdent, arguments); } /** @@ -1819,7 +1818,7 @@ loop: } } - return LiteralNode.newInstance(source, arrayToken, finish, elements); + return LiteralNode.newInstance(arrayToken, finish, elements); } /** @@ -1926,7 +1925,7 @@ loop: map.put(key, newProperty = newProperty.setValue(value)); } else { final long propertyToken = Token.recast(newProperty.getToken(), COMMARIGHT); - map.put(key, newProperty = newProperty.setValue(new BinaryNode(source, propertyToken, prevValue, value))); + map.put(key, newProperty = newProperty.setValue(new BinaryNode(propertyToken, prevValue, value))); } map.put(key, newProperty = newProperty.setGetter(null).setSetter(null)); @@ -1943,7 +1942,7 @@ loop: } } - return new ObjectNode(source, objectToken, finish, new ArrayList(map.values())); + return new ObjectNode(objectToken, finish, new ArrayList(map.values())); } /** @@ -2013,16 +2012,16 @@ loop: case "get": final PropertyKey getIdent = propertyName(); final String getterName = getIdent.getPropertyName(); - final IdentNode getNameNode = new IdentNode(source, ((Node)getIdent).getToken(), finish, "get " + getterName); + final IdentNode getNameNode = new IdentNode(((Node)getIdent).getToken(), finish, "get " + getterName); expect(LPAREN); expect(RPAREN); functionNode = functionBody(getSetToken, getNameNode, new ArrayList(), FunctionNode.Kind.GETTER); - return new PropertyNode(source, propertyToken, finish, getIdent, null, functionNode, null); + return new PropertyNode(propertyToken, finish, getIdent, null, functionNode, null); case "set": final PropertyKey setIdent = propertyName(); final String setterName = setIdent.getPropertyName(); - final IdentNode setNameNode = new IdentNode(source, ((Node)setIdent).getToken(), finish, "set " + setterName); + final IdentNode setNameNode = new IdentNode(((Node)setIdent).getToken(), finish, "set " + setterName); expect(LPAREN); final IdentNode argIdent = getIdent(); verifyStrictIdent(argIdent, "setter argument"); @@ -2030,21 +2029,21 @@ loop: List parameters = new ArrayList<>(); parameters.add(argIdent); functionNode = functionBody(getSetToken, setNameNode, parameters, FunctionNode.Kind.SETTER); - return new PropertyNode(source, propertyToken, finish, setIdent, null, null, functionNode); + return new PropertyNode(propertyToken, finish, setIdent, null, null, functionNode); default: break; } } - propertyName = new IdentNode(source, propertyToken, finish, ident); + propertyName = new IdentNode(propertyToken, finish, ident); } else { propertyName = propertyName(); } expect(COLON); - return new PropertyNode(source, propertyToken, finish, propertyName, assignmentExpression(false), null, null); + return new PropertyNode(propertyToken, finish, propertyName, assignmentExpression(false), null, null); } /** @@ -2076,7 +2075,7 @@ loop: detectSpecialFunction((IdentNode)lhs); } - lhs = new CallNode(source, callToken, finish, lhs, arguments); + lhs = new CallNode(callToken, finish, lhs, arguments); } loop: @@ -2090,7 +2089,7 @@ loop: final List arguments = argumentList(); // Create call node. - lhs = new CallNode(source, callToken, finish, lhs, arguments); + lhs = new CallNode(callToken, finish, lhs, arguments); break; @@ -2103,7 +2102,7 @@ loop: expect(RBRACKET); // Create indexing node. - lhs = new IndexNode(source, callToken, finish, lhs, rhs); + lhs = new IndexNode(callToken, finish, lhs, rhs); break; @@ -2113,7 +2112,7 @@ loop: final IdentNode property = getIdentifierName(); // Create property access node. - lhs = new AccessNode(source, callToken, finish, lhs, property); + lhs = new AccessNode(callToken, finish, lhs, property); break; @@ -2169,9 +2168,9 @@ loop: arguments.add(objectLiteral()); } - final CallNode callNode = new CallNode(source, constructor.getToken(), finish, constructor, arguments); + final CallNode callNode = new CallNode(constructor.getToken(), finish, constructor, arguments); - return new UnaryNode(source, newToken, callNode); + return new UnaryNode(newToken, callNode); } /** @@ -2223,7 +2222,7 @@ loop: expect(RBRACKET); // Create indexing node. - lhs = new IndexNode(source, callToken, finish, lhs, index); + lhs = new IndexNode(callToken, finish, lhs, index); break; @@ -2237,7 +2236,7 @@ loop: final IdentNode property = getIdentifierName(); // Create property access node. - lhs = new AccessNode(source, callToken, finish, lhs, property); + lhs = new AccessNode(callToken, finish, lhs, property); break; @@ -2326,7 +2325,7 @@ loop: boolean isAnonymous = false; if (name == null) { final String tmpName = "_L" + source.getLine(Token.descPosition(token)); - name = new IdentNode(source, functionToken, Token.descPosition(functionToken), tmpName); + name = new IdentNode(functionToken, Token.descPosition(functionToken), tmpName); isAnonymous = true; } @@ -2377,7 +2376,7 @@ loop: // rename in non-strict mode parameterName = functionNode.uniqueName(parameterName); final long parameterToken = parameter.getToken(); - parameters.set(i, new IdentNode(source, parameterToken, Token.descPosition(parameterToken), functionNode.uniqueName(parameterName))); + parameters.set(i, new IdentNode(parameterToken, Token.descPosition(parameterToken), functionNode.uniqueName(parameterName))); } parametersSet.add(parameterName); @@ -2389,7 +2388,7 @@ loop: } if (isStatement) { - final VarNode varNode = new VarNode(source, functionToken, finish, name, functionNode, VarNode.IS_STATEMENT); + final VarNode varNode = new VarNode(functionToken, finish, name, functionNode, VarNode.IS_STATEMENT); if (topLevel) { functionDeclarations.add(lineNumber); functionDeclarations.add(varNode); @@ -2469,7 +2468,7 @@ loop: assert lc.getCurrentBlock() == lc.getFunctionBody(functionNode); // create a return statement - this creates code in itself and does not need to be // wrapped into an ExecuteNode - final ReturnNode returnNode = new ReturnNode(source, expr.getToken(), finish, expr); + final ReturnNode returnNode = new ReturnNode(expr.getToken(), finish, expr); appendStatement(returnNode); lastToken = token; functionNode.setFinish(Token.descPosition(token) + Token.descLength(token)); @@ -2511,16 +2510,16 @@ loop: } } - private RuntimeNode referenceError(final Node lhs, final Node rhs) { + private static RuntimeNode referenceError(final Node lhs, final Node rhs) { final ArrayList args = new ArrayList<>(); args.add(lhs); if (rhs == null) { - args.add(LiteralNode.newInstance(source, lhs.getToken(), lhs.getFinish())); + args.add(LiteralNode.newInstance(lhs.getToken(), lhs.getFinish())); } else { args.add(rhs); } - args.add(LiteralNode.newInstance(source, lhs.getToken(), lhs.getFinish(), lhs.toString())); - return new RuntimeNode(source, lhs.getToken(), lhs.getFinish(), RuntimeNode.Request.REFERENCE_ERROR, args); + args.add(LiteralNode.newInstance(lhs.getToken(), lhs.getFinish(), lhs.toString())); + return new RuntimeNode(lhs.getToken(), lhs.getFinish(), RuntimeNode.Request.REFERENCE_ERROR, args); } /* @@ -2570,7 +2569,7 @@ loop: case NOT: next(); final Node expr = unaryExpression(); - return new UnaryNode(source, unaryToken, expr); + return new UnaryNode(unaryToken, expr); case INCPREFIX: case DECPREFIX: @@ -2759,7 +2758,7 @@ loop: final Node third = expression(unaryExpression(), ASSIGN.getPrecedence(), noIn); // Build up node. - lhs = new TernaryNode(source, op, lhs, rhs, third); + lhs = new TernaryNode(op, lhs, rhs, third); } else { // Skip operator. next(); @@ -2820,7 +2819,7 @@ loop: */ private LineNumberNode lineNumber() { if (env._debug_lines) { - return new LineNumberNode(source, token, line); + return new LineNumberNode(token, line); } return null; } diff --git a/nashorn/src/jdk/nashorn/internal/runtime/RecompilableScriptFunctionData.java b/nashorn/src/jdk/nashorn/internal/runtime/RecompilableScriptFunctionData.java index 1ba74f13db8..03a25dab1b7 100644 --- a/nashorn/src/jdk/nashorn/internal/runtime/RecompilableScriptFunctionData.java +++ b/nashorn/src/jdk/nashorn/internal/runtime/RecompilableScriptFunctionData.java @@ -229,6 +229,10 @@ public final class RecompilableScriptFunctionData extends ScriptFunctionData { } final FunctionNode snapshot = functionNode.getSnapshot(); + if (snapshot == null) { + return mh; + } + int i; //classes known at runtime diff --git a/nashorn/src/jdk/nashorn/internal/runtime/ScriptObject.java b/nashorn/src/jdk/nashorn/internal/runtime/ScriptObject.java index 01d61d7f510..61cce7b127a 100644 --- a/nashorn/src/jdk/nashorn/internal/runtime/ScriptObject.java +++ b/nashorn/src/jdk/nashorn/internal/runtime/ScriptObject.java @@ -970,9 +970,7 @@ public abstract class ScriptObject extends PropertyListenerManager implements Pr * @param bindName null or name to bind to second argument (property not found method.) * * @return value of property as a MethodHandle or null. - * */ - @SuppressWarnings("static-method") protected MethodHandle getCallMethodHandle(final FindProperty find, final MethodType type, final String bindName) { return getCallMethodHandle(getObjectValue(find), type, bindName); } diff --git a/nashorn/src/jdk/nashorn/internal/runtime/arrays/ArrayLikeIterator.java b/nashorn/src/jdk/nashorn/internal/runtime/arrays/ArrayLikeIterator.java index 1703b7ecf14..5061683a355 100644 --- a/nashorn/src/jdk/nashorn/internal/runtime/arrays/ArrayLikeIterator.java +++ b/nashorn/src/jdk/nashorn/internal/runtime/arrays/ArrayLikeIterator.java @@ -57,7 +57,6 @@ abstract public class ArrayLikeIterator implements Iterator { * Is this a reverse order iteration? * @return true if reverse */ - @SuppressWarnings("static-method") public boolean isReverse() { return false; } diff --git a/nashorn/src/jdk/nashorn/internal/runtime/linker/LinkerCallSite.java b/nashorn/src/jdk/nashorn/internal/runtime/linker/LinkerCallSite.java index 05b1a9396ef..ce71caada9d 100644 --- a/nashorn/src/jdk/nashorn/internal/runtime/linker/LinkerCallSite.java +++ b/nashorn/src/jdk/nashorn/internal/runtime/linker/LinkerCallSite.java @@ -275,7 +275,6 @@ public class LinkerCallSite extends ChainedCallSite { } static class ProfileDumper implements Runnable { - @SuppressWarnings("resource") @Override public void run() { PrintWriter out = null; @@ -447,7 +446,7 @@ public class LinkerCallSite extends ChainedCallSite { * * @throws Throwable if invocation fails or throws exception/error */ - @SuppressWarnings({"unused", "resource"}) + @SuppressWarnings("unused") public Object traceObject(final MethodHandle mh, final Object... args) throws Throwable { final PrintWriter out = Context.getCurrentErr(); tracePrint(out, "ENTER ", args, null); @@ -465,7 +464,7 @@ public class LinkerCallSite extends ChainedCallSite { * * @throws Throwable if invocation fails or throws exception/error */ - @SuppressWarnings({"unused", "resource"}) + @SuppressWarnings("unused") public void traceVoid(final MethodHandle mh, final Object... args) throws Throwable { final PrintWriter out = Context.getCurrentErr(); tracePrint(out, "ENTER ", args, null); diff --git a/nashorn/src/jdk/nashorn/tools/Shell.java b/nashorn/src/jdk/nashorn/tools/Shell.java index 708fecc59f1..ddc520e6b70 100644 --- a/nashorn/src/jdk/nashorn/tools/Shell.java +++ b/nashorn/src/jdk/nashorn/tools/Shell.java @@ -392,7 +392,6 @@ public class Shell { * @param global global scope object to use * @return return code */ - @SuppressWarnings("resource") private static int readEvalPrint(final Context context, final ScriptObject global) { final String prompt = bundle.getString("shell.prompt"); final BufferedReader in = new BufferedReader(new InputStreamReader(System.in)); From c668214703db7f1abe8501e401abd1c20893fcc1 Mon Sep 17 00:00:00 2001 From: Marcus Lagergren Date: Tue, 7 May 2013 14:43:17 +0200 Subject: [PATCH 11/15] 8013914: Removed explicit LineNumberNodes that were too brittle when code moves around, and also introduced unnecessary footprint. Introduced the Statement node and fixed dead code elimination issues that were discovered by the absense of labels for LineNumberNodes Reviewed-by: jlaskey, attila --- nashorn/make/project.properties | 2 +- .../jdk/nashorn/internal/codegen/Attr.java | 37 +++-- .../internal/codegen/CodeGenerator.java | 57 ++++++-- .../internal/codegen/FinalizeTypes.java | 6 +- .../internal/codegen/FoldConstants.java | 2 +- .../jdk/nashorn/internal/codegen/Label.java | 21 ++- .../jdk/nashorn/internal/codegen/Lower.java | 96 ++++++------- .../internal/codegen/MethodEmitter.java | 78 ++++++----- .../nashorn/internal/codegen/Splitter.java | 43 +++--- .../src/jdk/nashorn/internal/ir/Block.java | 20 +-- .../internal/ir/BlockLexicalContext.java | 32 ++--- .../jdk/nashorn/internal/ir/BreakNode.java | 13 +- .../nashorn/internal/ir/BreakableNode.java | 5 +- .../src/jdk/nashorn/internal/ir/CallNode.java | 13 +- .../jdk/nashorn/internal/ir/CatchNode.java | 7 +- .../jdk/nashorn/internal/ir/ContinueNode.java | 13 +- .../jdk/nashorn/internal/ir/EmptyNode.java | 9 +- .../jdk/nashorn/internal/ir/ExecuteNode.java | 17 +-- .../src/jdk/nashorn/internal/ir/ForNode.java | 19 +-- .../jdk/nashorn/internal/ir/FunctionNode.java | 19 ++- .../src/jdk/nashorn/internal/ir/IfNode.java | 17 +-- .../jdk/nashorn/internal/ir/LabelNode.java | 13 +- .../internal/ir/LexicalContextNode.java | 11 +- .../src/jdk/nashorn/internal/ir/LoopNode.java | 13 +- nashorn/src/jdk/nashorn/internal/ir/Node.java | 9 -- .../jdk/nashorn/internal/ir/ReturnNode.java | 11 +- .../jdk/nashorn/internal/ir/SplitNode.java | 5 +- .../{LineNumberNode.java => Statement.java} | 71 +++++----- .../jdk/nashorn/internal/ir/SwitchNode.java | 5 +- .../src/jdk/nashorn/internal/ir/Symbol.java | 20 +-- .../jdk/nashorn/internal/ir/ThrowNode.java | 9 +- .../src/jdk/nashorn/internal/ir/TryNode.java | 15 ++- .../src/jdk/nashorn/internal/ir/VarNode.java | 30 +++-- .../jdk/nashorn/internal/ir/WhileNode.java | 25 ++-- .../src/jdk/nashorn/internal/ir/WithNode.java | 5 +- .../nashorn/internal/ir/debug/JSONWriter.java | 34 ++--- .../internal/ir/debug/PrintVisitor.java | 50 +++---- .../ir/visitor/NodeOperatorVisitor.java | 2 +- .../internal/ir/visitor/NodeVisitor.java | 21 --- .../jdk/nashorn/internal/objects/Global.java | 1 + .../nashorn/internal/objects/NativeDebug.java | 1 + .../jdk/nashorn/internal/parser/Parser.java | 127 ++++++++++-------- .../jdk/nashorn/internal/runtime/Context.java | 3 +- .../runtime/linker/LinkerCallSite.java | 7 +- nashorn/src/jdk/nashorn/tools/Shell.java | 1 + nashorn/test/script/basic/no_line_numbers.js | 125 +++++++++++++++++ .../script/basic/no_line_numbers.js.EXPECTED | 12 ++ 47 files changed, 660 insertions(+), 492 deletions(-) rename nashorn/src/jdk/nashorn/internal/ir/{LineNumberNode.java => Statement.java} (58%) create mode 100644 nashorn/test/script/basic/no_line_numbers.js create mode 100644 nashorn/test/script/basic/no_line_numbers.js.EXPECTED diff --git a/nashorn/make/project.properties b/nashorn/make/project.properties index e2f39bb7ab0..bd4cf5d3890 100644 --- a/nashorn/make/project.properties +++ b/nashorn/make/project.properties @@ -214,7 +214,7 @@ run.test.xms=2G # -XX:+PrintCompilation -XX:+UnlockDiagnosticVMOptions -XX:+PrintNMethods # add '-Dtest.js.outofprocess' to run each test in a new sub-process -run.test.jvmargs.main=-server -Xmx${run.test.xmx} -XX:-TieredCompilation -ea -Dnashorn.debug=true -Dfile.encoding=UTF-8 +run.test.jvmargs.main=-server -Xmx${run.test.xmx} -XX:+TieredCompilation -ea -Dnashorn.debug=true -Dfile.encoding=UTF-8 #-XX:+HeapDumpOnOutOfMemoryError -XX:-UseCompressedKlassPointers -XX:+PrintHeapAtGC -XX:ClassMetaspaceSize=300M run.test.jvmargs.octane.main=-Xms${run.test.xms} ${run.test.jvmargs.main} diff --git a/nashorn/src/jdk/nashorn/internal/codegen/Attr.java b/nashorn/src/jdk/nashorn/internal/codegen/Attr.java index a381e58d7aa..8831f75794b 100644 --- a/nashorn/src/jdk/nashorn/internal/codegen/Attr.java +++ b/nashorn/src/jdk/nashorn/internal/codegen/Attr.java @@ -78,6 +78,7 @@ import jdk.nashorn.internal.ir.PropertyNode; import jdk.nashorn.internal.ir.ReturnNode; import jdk.nashorn.internal.ir.RuntimeNode; import jdk.nashorn.internal.ir.RuntimeNode.Request; +import jdk.nashorn.internal.ir.Statement; import jdk.nashorn.internal.ir.SwitchNode; import jdk.nashorn.internal.ir.Symbol; import jdk.nashorn.internal.ir.TernaryNode; @@ -154,7 +155,9 @@ final class Attr extends NodeOperatorVisitor { @Override public Node leaveAccessNode(final AccessNode accessNode) { - //While Object type is assigned here, Access Specialization in FinalizeTypes may narrow this + //While Object type is assigned here, Access Specialization in FinalizeTypes may narrow this, that + //is why we can't set the access node base to be an object here, that will ruin access specialization + //for example for a.x | 17. return end(ensureSymbol(Type.OBJECT, accessNode)); } @@ -435,6 +438,7 @@ final class Attr extends NodeOperatorVisitor { final IdentNode callee = compilerConstant(CALLEE); VarNode selfInit = new VarNode( + newFunctionNode.getLineNumber(), newFunctionNode.getToken(), newFunctionNode.getFinish(), newFunctionNode.getIdent(), @@ -442,7 +446,7 @@ final class Attr extends NodeOperatorVisitor { LOG.info("Accepting self symbol init ", selfInit, " for ", newFunctionNode.getName()); - final List newStatements = new ArrayList<>(); + final List newStatements = new ArrayList<>(); assert callee.getSymbol() != null && callee.getSymbol().hasSlot(); final IdentNode name = selfInit.getName(); @@ -492,7 +496,7 @@ final class Attr extends NodeOperatorVisitor { final Symbol pseudoSymbol = pseudoSymbol(name); LOG.info("IdentNode is property name -> assigning pseudo symbol ", pseudoSymbol); LOG.unindent(); - return identNode.setSymbol(lc, pseudoSymbol); + return end(identNode.setSymbol(lc, pseudoSymbol)); } final Block block = lc.getCurrentBlock(); @@ -658,7 +662,7 @@ final class Attr extends NodeOperatorVisitor { if (literalNode instanceof ArrayLiteralNode) { ((ArrayLiteralNode)literalNode).analyze(); } - return literalNode.setSymbol(getLexicalContext(), symbol); + return end(literalNode.setSymbol(getLexicalContext(), symbol)); } @Override @@ -779,17 +783,18 @@ final class Attr extends NodeOperatorVisitor { final IdentNode ident = newVarNode.getName(); final String name = ident.getName(); + final LexicalContext lc = getLexicalContext(); + final Symbol symbol = findSymbol(lc.getCurrentBlock(), ident.getName()); + if (init == null) { // var x; with no init will be treated like a use of x by // leaveIdentNode unless we remove the name from the localdef list. removeLocalDef(name); - return newVarNode; + return newVarNode.setSymbol(lc, symbol); } addLocalDef(name); - final LexicalContext lc = getLexicalContext(); - final Symbol symbol = findSymbol(lc.getCurrentBlock(), ident.getName()); assert symbol != null; final IdentNode newIdent = (IdentNode)ident.setSymbol(lc, symbol); @@ -1398,7 +1403,9 @@ final class Attr extends NodeOperatorVisitor { private FunctionNode finalizeParameters(final FunctionNode functionNode) { final List newParams = new ArrayList<>(); final boolean isVarArg = functionNode.isVarArg(); + final int nparams = functionNode.getParameters().size(); + int specialize = 0; int pos = 0; for (final IdentNode param : functionNode.getParameters()) { final Symbol paramSymbol = functionNode.getBody().getExistingSymbol(param.getName()); @@ -1414,9 +1421,13 @@ final class Attr extends NodeOperatorVisitor { // if we know that a parameter is only used as a certain type throughout // this function, we can tell the runtime system that no matter what the - // call site is, use this information. TODO - if (!paramSymbol.getSymbolType().isObject()) { + // call site is, use this information: + // we also need more than half of the parameters to be specializable + // for the heuristic to be worth it, and we need more than one use of + // the parameter to consider it, i.e. function(x) { call(x); } doens't count + if (paramSymbol.getUseCount() > 1 && !paramSymbol.getSymbolType().isObject()) { LOG.finest("Parameter ", param, " could profit from specialization to ", paramSymbol.getSymbolType()); + specialize++; } newType(paramSymbol, Type.widest(type, paramSymbol.getSymbolType())); @@ -1429,7 +1440,13 @@ final class Attr extends NodeOperatorVisitor { pos++; } - return functionNode.setParameters(getLexicalContext(), newParams); + FunctionNode newFunctionNode = functionNode; + + if (nparams == 0 || (specialize * 2) < nparams) { + newFunctionNode = newFunctionNode.clearSnapshot(getLexicalContext()); + } + + return newFunctionNode.setParameters(getLexicalContext(), newParams); } /** diff --git a/nashorn/src/jdk/nashorn/internal/codegen/CodeGenerator.java b/nashorn/src/jdk/nashorn/internal/codegen/CodeGenerator.java index 97219e3db8a..1fae129468d 100644 --- a/nashorn/src/jdk/nashorn/internal/codegen/CodeGenerator.java +++ b/nashorn/src/jdk/nashorn/internal/codegen/CodeGenerator.java @@ -63,6 +63,7 @@ import java.util.LinkedList; import java.util.List; import java.util.Map; import java.util.TreeMap; + import jdk.nashorn.internal.codegen.ClassEmitter.Flag; import jdk.nashorn.internal.codegen.CompilerConstants.Call; import jdk.nashorn.internal.codegen.RuntimeCallSite.SpecializedRuntimeNode; @@ -88,7 +89,6 @@ import jdk.nashorn.internal.ir.IfNode; import jdk.nashorn.internal.ir.IndexNode; import jdk.nashorn.internal.ir.LexicalContext; import jdk.nashorn.internal.ir.LexicalContextNode; -import jdk.nashorn.internal.ir.LineNumberNode; import jdk.nashorn.internal.ir.LiteralNode; import jdk.nashorn.internal.ir.LiteralNode.ArrayLiteralNode; import jdk.nashorn.internal.ir.LiteralNode.ArrayLiteralNode.ArrayUnit; @@ -100,6 +100,7 @@ import jdk.nashorn.internal.ir.ReturnNode; import jdk.nashorn.internal.ir.RuntimeNode; import jdk.nashorn.internal.ir.RuntimeNode.Request; import jdk.nashorn.internal.ir.SplitNode; +import jdk.nashorn.internal.ir.Statement; import jdk.nashorn.internal.ir.SwitchNode; import jdk.nashorn.internal.ir.Symbol; import jdk.nashorn.internal.ir.TernaryNode; @@ -191,6 +192,8 @@ final class CodeGenerator extends NodeOperatorVisitor { /** Current compile unit */ private CompileUnit unit; + private int lastLineNumber = -1; + /** When should we stop caching regexp expressions in fields to limit bytecode size? */ private static final int MAX_REGEX_FIELDS = 2 * 1024; @@ -619,6 +622,8 @@ final class CodeGenerator extends NodeOperatorVisitor { @Override public boolean enterBreakNode(final BreakNode breakNode) { + lineNumber(breakNode); + final BreakableNode breakFrom = getLexicalContext().getBreakable(breakNode.getLabel()); for (int i = 0; i < getLexicalContext().getScopeNestingLevelTo(breakFrom); i++) { closeWith(); @@ -663,6 +668,8 @@ final class CodeGenerator extends NodeOperatorVisitor { @Override public boolean enterCallNode(final CallNode callNode) { + lineNumber(callNode); + final List args = callNode.getArgs(); final Node function = callNode.getFunction(); final Block currentBlock = getLexicalContext().getCurrentBlock(); @@ -836,6 +843,8 @@ final class CodeGenerator extends NodeOperatorVisitor { @Override public boolean enterContinueNode(final ContinueNode continueNode) { + lineNumber(continueNode); + final LoopNode continueTo = getLexicalContext().getContinueTo(continueNode.getLabel()); for (int i = 0; i < getLexicalContext().getScopeNestingLevelTo(continueTo); i++) { closeWith(); @@ -847,11 +856,15 @@ final class CodeGenerator extends NodeOperatorVisitor { @Override public boolean enterEmptyNode(final EmptyNode emptyNode) { + lineNumber(emptyNode); + return false; } @Override public boolean enterExecuteNode(final ExecuteNode executeNode) { + lineNumber(executeNode); + final Node expression = executeNode.getExpression(); expression.accept(this); @@ -860,6 +873,8 @@ final class CodeGenerator extends NodeOperatorVisitor { @Override public boolean enterForNode(final ForNode forNode) { + lineNumber(forNode); + final Node test = forNode.getTest(); final Block body = forNode.getBody(); final Node modify = forNode.getModify(); @@ -1148,6 +1163,8 @@ final class CodeGenerator extends NodeOperatorVisitor { @Override public boolean enterIfNode(final IfNode ifNode) { + lineNumber(ifNode); + final Node test = ifNode.getTest(); final Block pass = ifNode.getPass(); final Block fail = ifNode.getFail(); @@ -1187,12 +1204,12 @@ final class CodeGenerator extends NodeOperatorVisitor { return false; } - @Override - public boolean enterLineNumberNode(final LineNumberNode lineNumberNode) { - final Label label = new Label((String)null); - method.label(label); - method.lineNumber(lineNumberNode.getLineNumber(), label); - return false; + private void lineNumber(final Statement statement) { + final int lineNumber = statement.getLineNumber(); + if (lineNumber != lastLineNumber) { + method.lineNumber(statement.getLineNumber()); + } + lastLineNumber = lineNumber; } /** @@ -1524,6 +1541,8 @@ final class CodeGenerator extends NodeOperatorVisitor { @Override public boolean enterReturnNode(final ReturnNode returnNode) { + lineNumber(returnNode); + method.registerReturn(); final Type returnType = getLexicalContext().getCurrentFunction().getReturnType(); @@ -1733,6 +1752,8 @@ final class CodeGenerator extends NodeOperatorVisitor { @Override public boolean enterSplitNode(final SplitNode splitNode) { + lineNumber(splitNode); + final CompileUnit splitCompileUnit = splitNode.getCompileUnit(); final FunctionNode fn = getLexicalContext().getCurrentFunction(); @@ -1876,6 +1897,8 @@ final class CodeGenerator extends NodeOperatorVisitor { @Override public boolean enterSwitchNode(final SwitchNode switchNode) { + lineNumber(switchNode); + final Node expression = switchNode.getExpression(); final Symbol tag = switchNode.getTag(); final boolean allInteger = tag.getSymbolType().isInteger(); @@ -1958,7 +1981,6 @@ final class CodeGenerator extends NodeOperatorVisitor { method.tableswitch(lo, hi, defaultLabel, table); } else { final int[] ints = new int[size]; - for (int i = 0; i < size; i++) { ints[i] = values[i]; } @@ -2004,6 +2026,8 @@ final class CodeGenerator extends NodeOperatorVisitor { @Override public boolean enterThrowNode(final ThrowNode throwNode) { + lineNumber(throwNode); + method._new(ECMAException.class).dup(); final Source source = getLexicalContext().getCurrentFunction().getSource(); @@ -2028,6 +2052,8 @@ final class CodeGenerator extends NodeOperatorVisitor { @Override public boolean enterTryNode(final TryNode tryNode) { + lineNumber(tryNode); + final Block body = tryNode.getBody(); final List catchBlocks = tryNode.getCatchBlocks(); final Symbol symbol = tryNode.getException(); @@ -2124,12 +2150,15 @@ final class CodeGenerator extends NodeOperatorVisitor { @Override public boolean enterVarNode(final VarNode varNode) { + final Node init = varNode.getInit(); if (init == null) { return false; } + lineNumber(varNode); + final Symbol varSymbol = varNode.getSymbol(); assert varSymbol != null : "variable node " + varNode + " requires a symbol"; @@ -2162,6 +2191,8 @@ final class CodeGenerator extends NodeOperatorVisitor { @Override public boolean enterWhileNode(final WhileNode whileNode) { + lineNumber(whileNode); + final Node test = whileNode.getTest(); final Block body = whileNode.getBody(); final Label breakLabel = whileNode.getBreakLabel(); @@ -2184,7 +2215,7 @@ final class CodeGenerator extends NodeOperatorVisitor { } private void closeWith() { - if(method.hasScope()) { + if (method.hasScope()) { method.loadCompilerConstant(SCOPE); method.invoke(ScriptRuntime.CLOSE_WITH); method.storeCompilerConstant(SCOPE); @@ -2227,7 +2258,7 @@ final class CodeGenerator extends NodeOperatorVisitor { // Always process body body.accept(this); - if(hasScope) { + if (hasScope) { // Ensure we always close the WithObject final Label endLabel = new Label("with_end"); final Label catchLabel = new Label("with_catch"); @@ -3010,6 +3041,7 @@ final class CodeGenerator extends NodeOperatorVisitor { * @param block the block we are in * @param ident identifier for block or function where applicable */ + @SuppressWarnings("resource") private void printSymbols(final Block block, final String ident) { if (!compiler.getEnv()._print_symbols) { return; @@ -3190,9 +3222,6 @@ final class CodeGenerator extends NodeOperatorVisitor { return; } - //System.err.println("Store with out discard that shouldn't just return " + assignNode); - //new Throwable().printStackTrace(); - final Symbol symbol = assignNode.getSymbol(); if (symbol.hasSlot()) { method.dup().store(symbol); @@ -3288,7 +3317,7 @@ final class CodeGenerator extends NodeOperatorVisitor { // Such immediately-called functions are invoked using INVOKESTATIC (see enterFunctionNode() of the embedded // visitor of enterCallNode() for details), and if they don't need a callee, they don't have it on their // static method's parameter list. - if(lc.getOutermostFunction() == functionNode || + if (lc.getOutermostFunction() == functionNode || (!functionNode.needsCallee()) && lc.isFunctionDefinedInCurrentCall(originalFunctionNode)) { return; } diff --git a/nashorn/src/jdk/nashorn/internal/codegen/FinalizeTypes.java b/nashorn/src/jdk/nashorn/internal/codegen/FinalizeTypes.java index 1b6e3c161df..e7ac516b91c 100644 --- a/nashorn/src/jdk/nashorn/internal/codegen/FinalizeTypes.java +++ b/nashorn/src/jdk/nashorn/internal/codegen/FinalizeTypes.java @@ -228,19 +228,19 @@ final class FinalizeTypes extends NodeOperatorVisitor { } @Override - public Node leaveBIT_AND(BinaryNode binaryNode) { + public Node leaveBIT_AND(final BinaryNode binaryNode) { assert binaryNode.getSymbol() != null && binaryNode.getSymbol().getSymbolType().isInteger() : "int coercion expected: " + binaryNode.getSymbol(); return leaveBinary(binaryNode, Type.INT, Type.INT); } @Override - public Node leaveBIT_OR(BinaryNode binaryNode) { + public Node leaveBIT_OR(final BinaryNode binaryNode) { assert binaryNode.getSymbol() != null && binaryNode.getSymbol().getSymbolType().isInteger() : "int coercion expected: " + binaryNode.getSymbol(); return leaveBinary(binaryNode, Type.INT, Type.INT); } @Override - public Node leaveBIT_XOR(BinaryNode binaryNode) { + public Node leaveBIT_XOR(final BinaryNode binaryNode) { assert binaryNode.getSymbol() != null && binaryNode.getSymbol().getSymbolType().isInteger() : "int coercion expected: " + binaryNode.getSymbol(); return leaveBinary(binaryNode, Type.INT, Type.INT); } diff --git a/nashorn/src/jdk/nashorn/internal/codegen/FoldConstants.java b/nashorn/src/jdk/nashorn/internal/codegen/FoldConstants.java index 92286c2bf63..686c9836da6 100644 --- a/nashorn/src/jdk/nashorn/internal/codegen/FoldConstants.java +++ b/nashorn/src/jdk/nashorn/internal/codegen/FoldConstants.java @@ -88,7 +88,7 @@ final class FoldConstants extends NodeVisitor { if (test instanceof LiteralNode) { final Block shortCut = ((LiteralNode)test).isTrue() ? ifNode.getPass() : ifNode.getFail(); if (shortCut != null) { - return new ExecuteNode(shortCut); + return new ExecuteNode(shortCut.getLineNumber(), shortCut.getToken(), shortCut.getFinish(), shortCut); } return new EmptyNode(ifNode); } diff --git a/nashorn/src/jdk/nashorn/internal/codegen/Label.java b/nashorn/src/jdk/nashorn/internal/codegen/Label.java index 9d28caafd1d..4c401370041 100644 --- a/nashorn/src/jdk/nashorn/internal/codegen/Label.java +++ b/nashorn/src/jdk/nashorn/internal/codegen/Label.java @@ -27,6 +27,7 @@ package jdk.nashorn.internal.codegen; import java.util.ArrayDeque; import jdk.nashorn.internal.codegen.types.Type; +import jdk.nashorn.internal.runtime.Debug; /** * Abstraction for labels, separating a label from the underlying @@ -35,13 +36,16 @@ import jdk.nashorn.internal.codegen.types.Type; * * see -Dnashorn.codegen.debug, --log=codegen */ -public class Label extends jdk.internal.org.objectweb.asm.Label { +public final class Label { /** Name of this label */ private final String name; /** Type stack at this label */ private ArrayDeque stack; + /** ASM representation of this label */ + private jdk.internal.org.objectweb.asm.Label label; + /** * Constructor * @@ -62,6 +66,14 @@ public class Label extends jdk.internal.org.objectweb.asm.Label { this.name = label.name; } + + jdk.internal.org.objectweb.asm.Label getLabel() { + if (this.label == null) { + this.label = new jdk.internal.org.objectweb.asm.Label(); + } + return label; + } + ArrayDeque getStack() { return stack; } @@ -72,12 +84,7 @@ public class Label extends jdk.internal.org.objectweb.asm.Label { @Override public String toString() { - final StringBuilder sb = new StringBuilder(); - String s = super.toString(); - s = s.substring(1, s.length()); - sb.append(name).append('_').append(Long.toHexString(Long.parseLong(s))); - - return sb.toString(); + return name + '_' + Debug.id(this); } } diff --git a/nashorn/src/jdk/nashorn/internal/codegen/Lower.java b/nashorn/src/jdk/nashorn/internal/codegen/Lower.java index 6f55841ebc9..23e91b63026 100644 --- a/nashorn/src/jdk/nashorn/internal/codegen/Lower.java +++ b/nashorn/src/jdk/nashorn/internal/codegen/Lower.java @@ -32,6 +32,7 @@ import static jdk.nashorn.internal.codegen.CompilerConstants.THIS; import java.util.ArrayList; import java.util.Arrays; import java.util.List; + import jdk.nashorn.internal.ir.BaseNode; import jdk.nashorn.internal.ir.BinaryNode; import jdk.nashorn.internal.ir.Block; @@ -49,16 +50,15 @@ import jdk.nashorn.internal.ir.IdentNode; import jdk.nashorn.internal.ir.IfNode; import jdk.nashorn.internal.ir.LabelNode; import jdk.nashorn.internal.ir.LexicalContext; -import jdk.nashorn.internal.ir.LineNumberNode; import jdk.nashorn.internal.ir.LiteralNode; import jdk.nashorn.internal.ir.LoopNode; import jdk.nashorn.internal.ir.Node; import jdk.nashorn.internal.ir.ReturnNode; +import jdk.nashorn.internal.ir.Statement; import jdk.nashorn.internal.ir.SwitchNode; import jdk.nashorn.internal.ir.Symbol; import jdk.nashorn.internal.ir.ThrowNode; import jdk.nashorn.internal.ir.TryNode; -import jdk.nashorn.internal.ir.UnaryNode; import jdk.nashorn.internal.ir.VarNode; import jdk.nashorn.internal.ir.WhileNode; import jdk.nashorn.internal.ir.WithNode; @@ -93,21 +93,25 @@ final class Lower extends NodeOperatorVisitor { super(new BlockLexicalContext() { @Override - public List popStatements() { - List newStatements = new ArrayList<>(); + public List popStatements() { + final List newStatements = new ArrayList<>(); boolean terminated = false; - final List statements = super.popStatements(); - for (final Node statement : statements) { + final List statements = super.popStatements(); + for (final Statement statement : statements) { if (!terminated) { newStatements.add(statement); - if (statement.isTerminal()) { + if (statement.isTerminal() || statement instanceof BreakNode || statement instanceof ContinueNode) { //TODO hasGoto? But some Loops are hasGoto too - why? terminated = true; } } else { - if (statement instanceof VarNode) { - newStatements.add(((VarNode)statement).setInit(null)); - } + statement.accept(new NodeVisitor() { + @Override + public boolean enterVarNode(final VarNode varNode) { + newStatements.add(varNode.setInit(null)); + return false; + } + }); } } return newStatements; @@ -120,7 +124,7 @@ final class Lower extends NodeOperatorVisitor { final LexicalContext lc = getLexicalContext(); final FunctionNode function = lc.getCurrentFunction(); if (lc.isFunctionBody() && function.isProgram() && !function.hasDeclaredFunctions()) { - new ExecuteNode(block.getToken(), block.getFinish(), LiteralNode.newInstance(block, ScriptRuntime.UNDEFINED)).accept(this); + new ExecuteNode(block.getLineNumber(), block.getToken(), block.getFinish(), LiteralNode.newInstance(block, ScriptRuntime.UNDEFINED)).accept(this); } return true; } @@ -132,19 +136,20 @@ final class Lower extends NodeOperatorVisitor { final BlockLexicalContext lc = (BlockLexicalContext)getLexicalContext(); - Node last = lc.getLastStatement(); + Statement last = lc.getLastStatement(); if (lc.isFunctionBody()) { final FunctionNode currentFunction = getLexicalContext().getCurrentFunction(); final boolean isProgram = currentFunction.isProgram(); final ReturnNode returnNode = new ReturnNode( + last == null ? block.getLineNumber() : last.getLineNumber(), //TODO? currentFunction.getToken(), currentFunction.getFinish(), isProgram ? compilerConstant(RETURN) : LiteralNode.newInstance(block, ScriptRuntime.UNDEFINED)); - last = returnNode.accept(this); + last = (Statement)returnNode.accept(this); } if (last != null && last.isTerminal()) { @@ -238,12 +243,6 @@ final class Lower extends NodeOperatorVisitor { return addStatement(labelNode); } - @Override - public boolean enterLineNumberNode(final LineNumberNode lineNumberNode) { - addStatement(lineNumberNode); // don't put it in lastStatement cache - return false; - } - @Override public Node leaveReturnNode(final ReturnNode returnNode) { addStatement(returnNode); //ReturnNodes are always terminal, marked as such in constructor @@ -271,10 +270,10 @@ final class Lower extends NodeOperatorVisitor { }); } - private static List copyFinally(final Block finallyBody) { - final List newStatements = new ArrayList<>(); - for (final Node statement : finallyBody.getStatements()) { - newStatements.add(ensureUniqueLabelsIn(statement)); + private static List copyFinally(final Block finallyBody) { + final List newStatements = new ArrayList<>(); + for (final Statement statement : finallyBody.getStatements()) { + newStatements.add((Statement)ensureUniqueLabelsIn(statement)); if (statement.hasTerminalFlags()) { return newStatements; } @@ -283,16 +282,17 @@ final class Lower extends NodeOperatorVisitor { } private Block catchAllBlock(final TryNode tryNode) { - final long token = tryNode.getToken(); - final int finish = tryNode.getFinish(); + final int lineNumber = tryNode.getLineNumber(); + final long token = tryNode.getToken(); + final int finish = tryNode.getFinish(); final IdentNode exception = new IdentNode(token, finish, getLexicalContext().getCurrentFunction().uniqueName("catch_all")); - final Block catchBody = new Block(token, finish, new ThrowNode(token, finish, new IdentNode(exception))). + final Block catchBody = new Block(lineNumber, token, finish, new ThrowNode(lineNumber, token, finish, new IdentNode(exception))). setIsTerminal(getLexicalContext(), true); //ends with throw, so terminal - final CatchNode catchAllNode = new CatchNode(token, finish, new IdentNode(exception), null, catchBody); - final Block catchAllBlock = new Block(token, finish, catchAllNode); + final CatchNode catchAllNode = new CatchNode(lineNumber, token, finish, new IdentNode(exception), null, catchBody); + final Block catchAllBlock = new Block(lineNumber, token, finish, catchAllNode); //catchallblock -> catchallnode (catchnode) -> exception -> throw @@ -304,7 +304,7 @@ final class Lower extends NodeOperatorVisitor { return new IdentNode(functionNode.getToken(), functionNode.getFinish(), cc.symbolName()); } - private static boolean isTerminal(final List statements) { + private static boolean isTerminal(final List statements) { return !statements.isEmpty() && statements.get(statements.size() - 1).hasTerminalFlags(); } @@ -316,7 +316,7 @@ final class Lower extends NodeOperatorVisitor { * @return new try node after splicing finally code (same if nop) */ private Node spliceFinally(final TryNode tryNode, final List rethrows, final Block finallyBody) { - final int finish = tryNode.getFinish(); + final int finish = tryNode.getFinish(); assert tryNode.getFinallyBody() == null; @@ -338,11 +338,11 @@ final class Lower extends NodeOperatorVisitor { @Override public Node leaveThrowNode(final ThrowNode throwNode) { if (rethrows.contains(throwNode)) { - final List newStatements = copyFinally(finallyBody); + final List newStatements = copyFinally(finallyBody); if (!isTerminal(newStatements)) { newStatements.add(throwNode); } - return new Block(throwNode.getToken(), throwNode.getFinish(), newStatements); + return new Block(throwNode.getLineNumber(), throwNode.getToken(), throwNode.getFinish(), newStatements); } return throwNode; } @@ -360,14 +360,14 @@ final class Lower extends NodeOperatorVisitor { @Override public Node leaveReturnNode(final ReturnNode returnNode) { final Node expr = returnNode.getExpression(); - final List newStatements = new ArrayList<>(); + final List newStatements = new ArrayList<>(); final Node resultNode; if (expr != null) { //we need to evaluate the result of the return in case it is complex while //still in the try block, store it in a result value and return it afterwards resultNode = new IdentNode(Lower.this.compilerConstant(RETURN)); - newStatements.add(new ExecuteNode(new BinaryNode(Token.recast(returnNode.getToken(), TokenType.ASSIGN), resultNode, expr))); + newStatements.add(new ExecuteNode(returnNode.getLineNumber(), returnNode.getToken(), returnNode.getFinish(), new BinaryNode(Token.recast(returnNode.getToken(), TokenType.ASSIGN), resultNode, expr))); } else { resultNode = null; } @@ -377,16 +377,16 @@ final class Lower extends NodeOperatorVisitor { newStatements.add(expr == null ? returnNode : returnNode.setExpression(resultNode)); } - return new ExecuteNode(new Block(returnNode.getToken(), getLexicalContext().getCurrentBlock().getFinish(), newStatements)); + return new ExecuteNode(returnNode.getLineNumber(), returnNode.getToken(), returnNode.getFinish(), new Block(returnNode.getLineNumber(), returnNode.getToken(), getLexicalContext().getCurrentBlock().getFinish(), newStatements)); } - private Node copy(final Node endpoint, final Node targetNode) { + private Node copy(final Statement endpoint, final Node targetNode) { if (!insideTry.contains(targetNode)) { - final List newStatements = copyFinally(finallyBody); + final List newStatements = copyFinally(finallyBody); if (!isTerminal(newStatements)) { newStatements.add(endpoint); } - return new ExecuteNode(new Block(endpoint.getToken(), finish, newStatements)); + return new ExecuteNode(endpoint.getLineNumber(), endpoint.getToken(), endpoint.getFinish(), new Block(endpoint.getLineNumber(), endpoint.getToken(), finish, newStatements)); } return endpoint; } @@ -394,7 +394,7 @@ final class Lower extends NodeOperatorVisitor { addStatement(newTryNode); for (final Node statement : finallyBody.getStatements()) { - addStatement(statement); + addStatement((Statement)statement); } return newTryNode; @@ -448,7 +448,7 @@ final class Lower extends NodeOperatorVisitor { if (tryNode.getCatchBlocks().isEmpty()) { newTryNode = tryNode.setFinallyBody(null); } else { - Block outerBody = new Block(tryNode.getToken(), tryNode.getFinish(), new ArrayList(Arrays.asList(tryNode.setFinallyBody(null)))); + Block outerBody = new Block(tryNode.getLineNumber(), tryNode.getToken(), tryNode.getFinish(), new ArrayList(Arrays.asList(tryNode.setFinallyBody(null)))); newTryNode = tryNode.setBody(outerBody).setCatchBlocks(null); } @@ -465,7 +465,7 @@ final class Lower extends NodeOperatorVisitor { public Node leaveVarNode(final VarNode varNode) { addStatement(varNode); if (varNode.getFlag(VarNode.IS_LAST_FUNCTION_DECLARATION) && getLexicalContext().getCurrentFunction().isProgram()) { - new ExecuteNode(varNode.getToken(), varNode.getFinish(), new IdentNode(varNode.getName())).accept(this); + new ExecuteNode(varNode.getLineNumber(), varNode.getToken(), varNode.getFinish(), new IdentNode(varNode.getName())).accept(this); } return varNode; } @@ -477,7 +477,7 @@ final class Lower extends NodeOperatorVisitor { if (conservativeAlwaysTrue(test)) { //turn it into a for node without a test. - final ForNode forNode = (ForNode)new ForNode(whileNode.getToken(), whileNode.getFinish(), null, null, body, null, ForNode.IS_FOR).accept(this); + final ForNode forNode = (ForNode)new ForNode(whileNode.getLineNumber(), whileNode.getToken(), whileNode.getFinish(), null, null, body, null, ForNode.IS_FOR).accept(this); getLexicalContext().replace(whileNode, forNode); return forNode; } @@ -490,16 +490,6 @@ final class Lower extends NodeOperatorVisitor { return addStatement(withNode); } - @Override - public Node leaveDELETE(final UnaryNode unaryNode) { - final Node rhs = unaryNode.rhs(); - if (rhs instanceof IdentNode || rhs instanceof BaseNode) { - return unaryNode; - } - addStatement(new ExecuteNode(rhs)); - return LiteralNode.newInstance(unaryNode, true); - } - /** * Given a function node that is a callee in a CallNode, replace it with * the appropriate marker function. This is used by {@link CodeGenerator} @@ -616,7 +606,7 @@ final class Lower extends NodeOperatorVisitor { } - private Node addStatement(final Node statement) { + private Node addStatement(final Statement statement) { ((BlockLexicalContext)getLexicalContext()).appendStatement(statement); return statement; } diff --git a/nashorn/src/jdk/nashorn/internal/codegen/MethodEmitter.java b/nashorn/src/jdk/nashorn/internal/codegen/MethodEmitter.java index 4fbb57a68cf..549e264899f 100644 --- a/nashorn/src/jdk/nashorn/internal/codegen/MethodEmitter.java +++ b/nashorn/src/jdk/nashorn/internal/codegen/MethodEmitter.java @@ -71,6 +71,7 @@ import java.util.ArrayDeque; import java.util.EnumSet; import java.util.Iterator; import java.util.List; + import jdk.internal.dynalink.support.NameCodec; import jdk.internal.org.objectweb.asm.Handle; import jdk.internal.org.objectweb.asm.MethodVisitor; @@ -189,7 +190,7 @@ public class MethodEmitter implements Emitter { @Override public void begin() { classEmitter.beginMethod(this); - stack = new ArrayDeque<>(); + newStack(); method.visitCode(); } @@ -205,6 +206,10 @@ public class MethodEmitter implements Emitter { classEmitter.endMethod(this); } + private void newStack() { + stack = new ArrayDeque<>(); + } + @Override public String toString() { return "methodEmitter: " + (functionNode == null ? method : functionNode.getName()).toString() + ' ' + Debug.id(this); @@ -484,7 +489,7 @@ public class MethodEmitter implements Emitter { name = THIS_DEBUGGER.symbolName(); } - method.visitLocalVariable(name, symbol.getSymbolType().getDescriptor(), null, start, end, symbol.getSlot()); + method.visitLocalVariable(name, symbol.getSymbolType().getDescriptor(), null, start.getLabel(), end.getLabel(), symbol.getSlot()); } /** @@ -508,17 +513,6 @@ public class MethodEmitter implements Emitter { return invoke(virtualCallNoLookup(StringBuilder.class, "append", StringBuilder.class, String.class)); } - /** - * Associate a variable with a given range - * - * @param name name of the variable - * @param start start - * @param end end - */ - void markerVariable(final String name, final Label start, final Label end) { - method.visitLocalVariable(name, Type.OBJECT.getDescriptor(), null, start, end, 0); - } - /** * Pops two integer types from the stack, performs a bitwise and and pushes * the result @@ -626,7 +620,7 @@ public class MethodEmitter implements Emitter { * @param typeDescriptor type descriptor for exception */ void _try(final Label entry, final Label exit, final Label recovery, final String typeDescriptor) { - method.visitTryCatchBlock(entry, exit, recovery, typeDescriptor); + method.visitTryCatchBlock(entry.getLabel(), exit.getLabel(), recovery.getLabel(), typeDescriptor); } /** @@ -638,7 +632,7 @@ public class MethodEmitter implements Emitter { * @param clazz exception class */ void _try(final Label entry, final Label exit, final Label recovery, final Class clazz) { - method.visitTryCatchBlock(entry, exit, recovery, CompilerConstants.className(clazz)); + method.visitTryCatchBlock(entry.getLabel(), exit.getLabel(), recovery.getLabel(), CompilerConstants.className(clazz)); } /** @@ -1228,6 +1222,14 @@ public class MethodEmitter implements Emitter { return invoke(INVOKEINTERFACE, className, methodName, methodDescriptor, true); } + static jdk.internal.org.objectweb.asm.Label[] getLabels(final Label... table) { + final jdk.internal.org.objectweb.asm.Label[] internalLabels = new jdk.internal.org.objectweb.asm.Label[table.length]; + for (int i = 0; i < table.length; i++) { + internalLabels[i] = table[i].getLabel(); + } + return internalLabels; + } + /** * Generate a lookup switch, popping the switch value from the stack * @@ -1235,10 +1237,10 @@ public class MethodEmitter implements Emitter { * @param values case values for the table * @param table default label */ - void lookupswitch(final Label defaultLabel, final int[] values, final Label[] table) { + void lookupswitch(final Label defaultLabel, final int[] values, final Label... table) {//Collection