diff options
Diffstat (limited to 'src/jdk/nashorn/internal/runtime/arrays/ContinuousArrayData.java')
-rw-r--r-- | src/jdk/nashorn/internal/runtime/arrays/ContinuousArrayData.java | 364 |
1 files changed, 364 insertions, 0 deletions
diff --git a/src/jdk/nashorn/internal/runtime/arrays/ContinuousArrayData.java b/src/jdk/nashorn/internal/runtime/arrays/ContinuousArrayData.java new file mode 100644 index 00000000..2fdd184e --- /dev/null +++ b/src/jdk/nashorn/internal/runtime/arrays/ContinuousArrayData.java @@ -0,0 +1,364 @@ +/* + * 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.arrays; + +import static jdk.nashorn.internal.codegen.CompilerConstants.staticCall; +import static jdk.nashorn.internal.lookup.Lookup.MH; +import static jdk.nashorn.internal.runtime.JSType.getAccessorTypeIndex; +import static jdk.nashorn.internal.runtime.UnwarrantedOptimismException.INVALID_PROGRAM_POINT; +import static jdk.nashorn.internal.runtime.UnwarrantedOptimismException.isValid; +import java.lang.invoke.MethodHandle; +import java.lang.invoke.MethodHandles; +import java.lang.invoke.MethodType; +import java.lang.invoke.SwitchPoint; +import jdk.internal.dynalink.CallSiteDescriptor; +import jdk.internal.dynalink.linker.GuardedInvocation; +import jdk.internal.dynalink.linker.LinkRequest; +import jdk.nashorn.internal.codegen.types.Type; +import jdk.nashorn.internal.lookup.Lookup; +import jdk.nashorn.internal.runtime.ScriptObject; +import jdk.nashorn.internal.runtime.linker.NashornCallSiteDescriptor; +import jdk.nashorn.internal.runtime.logging.Logger; + +/** + * Interface implemented by all arrays that are directly accessible as underlying + * native arrays + */ +@Logger(name="arrays") +public abstract class ContinuousArrayData extends ArrayData { + /** + * Constructor + * @param length length (elementLength) + */ + protected ContinuousArrayData(final long length) { + super(length); + } + + /** + * Check if we can put one more element at the end of this continous + * array without reallocating, or if we are overwriting an already + * allocated element + * + * @param index index to check + * @return true if we don't need to do any array reallocation to fit an element at index + */ + public final boolean hasRoomFor(final int index) { + return has(index) || (index == length() && ensure(index) == this); + } + + /** + * Check if an arraydata is empty + * @return true if empty + */ + public boolean isEmpty() { + return length() == 0L; + } + + /** + * Return element getter for a certain type at a certain program point + * @param returnType return type + * @param programPoint program point + * @return element getter or null if not supported (used to implement slow linkage instead + * as fast isn't possible) + */ + public abstract MethodHandle getElementGetter(final Class<?> returnType, final int programPoint); + + /** + * Return element getter for a certain type at a certain program point + * @param elementType element type + * @return element setter or null if not supported (used to implement slow linkage instead + * as fast isn't possible) + */ + public abstract MethodHandle getElementSetter(final Class<?> elementType); + + /** + * Version of has that throws a class cast exception if element does not exist + * used for relinking + * + * @param index index to check - currently only int indexes + * @return index + */ + protected final int throwHas(final int index) { + if (!has(index)) { + throw new ClassCastException(); + } + return index; + } + + @Override + public abstract ContinuousArrayData copy(); + + /** + * Returns the type used to store an element in this array + * @return element type + */ + public abstract Class<?> getElementType(); + + @Override + public Type getOptimisticType() { + return Type.typeFor(getElementType()); + } + + /** + * Returns the boxed type of the type used to store an element in this array + * @return element type + */ + public abstract Class<?> getBoxedElementType(); + + /** + * Get the widest element type of two arrays. This can be done faster in subclasses, but + * this works for all ContinuousArrayDatas and for where more optimal checks haven't been + * implemented. + * + * @param otherData another ContinuousArrayData + * @return the widest boxed element type + */ + public ContinuousArrayData widest(final ContinuousArrayData otherData) { + final Class<?> elementType = getElementType(); + return Type.widest(elementType, otherData.getElementType()) == elementType ? this : otherData; + } + + /** + * Look up a continuous array element getter + * @param get getter, sometimes combined with a has check that throws CCE on failure for relink + * @param returnType return type + * @param programPoint program point + * @return array getter + */ + protected final MethodHandle getContinuousElementGetter(final MethodHandle get, final Class<?> returnType, final int programPoint) { + return getContinuousElementGetter(getClass(), get, returnType, programPoint); + } + + /** + * Look up a continuous array element setter + * @param set setter, sometimes combined with a has check that throws CCE on failure for relink + * @param returnType return type + * @return array setter + */ + protected final MethodHandle getContinuousElementSetter(final MethodHandle set, final Class<?> returnType) { + return getContinuousElementSetter(getClass(), set, returnType); + } + + /** + * Return element getter for a {@link ContinuousArrayData} + * @param clazz clazz for exact type guard + * @param getHas has getter + * @param returnType return type + * @param programPoint program point + * @return method handle for element setter + */ + protected MethodHandle getContinuousElementGetter(final Class<? extends ContinuousArrayData> clazz, final MethodHandle getHas, final Class<?> returnType, final int programPoint) { + final boolean isOptimistic = isValid(programPoint); + final int fti = getAccessorTypeIndex(getHas.type().returnType()); + final int ti = getAccessorTypeIndex(returnType); + MethodHandle mh = getHas; + + if (isOptimistic) { + if (ti < fti) { + mh = MH.insertArguments(ArrayData.THROW_UNWARRANTED.methodHandle(), 1, programPoint); + } + } + mh = MH.asType(mh, mh.type().changeReturnType(returnType).changeParameterType(0, clazz)); + + if (!isOptimistic) { + //for example a & array[17]; + return Lookup.filterReturnType(mh, returnType); + } + return mh; + } + + /** + * Return element setter for a {@link ContinuousArrayData} + * @param clazz clazz for exact type guard + * @param setHas set has guard + * @param elementType element type + * @return method handle for element setter + */ + protected MethodHandle getContinuousElementSetter(final Class<? extends ContinuousArrayData> clazz, final MethodHandle setHas, final Class<?> elementType) { + return MH.asType(setHas, setHas.type().changeParameterType(2, elementType).changeParameterType(0, clazz)); + } + + /** Fast access guard - it is impractical for JIT performance reasons to use only CCE asType as guard :-(, also we need + the null case explicitly, which is the one that CCE doesn't handle */ + protected static final MethodHandle FAST_ACCESS_GUARD = + MH.dropArguments( + staticCall( + MethodHandles.lookup(), + ContinuousArrayData.class, + "guard", + boolean.class, + Class.class, + ScriptObject.class).methodHandle(), + 2, + int.class); + + @SuppressWarnings("unused") + private static final boolean guard(final Class<? extends ContinuousArrayData> clazz, final ScriptObject sobj) { + if (sobj != null && sobj.getArray().getClass() == clazz) { + return true; + } + return false; + } + + /** + * Return a fast linked array getter, or null if we have to dispatch to super class + * @param desc descriptor + * @param request link request + * @return invocation or null if needs to be sent to slow relink + */ + @Override + public GuardedInvocation findFastGetIndexMethod(final Class<? extends ArrayData> clazz, final CallSiteDescriptor desc, final LinkRequest request) { + final MethodType callType = desc.getMethodType(); + final Class<?> indexType = callType.parameterType(1); + final Class<?> returnType = callType.returnType(); + + if (ContinuousArrayData.class.isAssignableFrom(clazz) && indexType == int.class) { + final Object[] args = request.getArguments(); + final int index = (int)args[args.length - 1]; + + if (has(index)) { + final MethodHandle getArray = ScriptObject.GET_ARRAY.methodHandle(); + final int programPoint = NashornCallSiteDescriptor.isOptimistic(desc) ? NashornCallSiteDescriptor.getProgramPoint(desc) : INVALID_PROGRAM_POINT; + MethodHandle getElement = getElementGetter(returnType, programPoint); + if (getElement != null) { + getElement = MH.filterArguments(getElement, 0, MH.asType(getArray, getArray.type().changeReturnType(clazz))); + final MethodHandle guard = MH.insertArguments(FAST_ACCESS_GUARD, 0, clazz); + return new GuardedInvocation(getElement, guard, (SwitchPoint)null, ClassCastException.class); + } + } + } + + return null; + } + + /** + * Return a fast linked array setter, or null if we have to dispatch to super class + * @param desc descriptor + * @param request link request + * @return invocation or null if needs to be sent to slow relink + */ + @Override + public GuardedInvocation findFastSetIndexMethod(final Class<? extends ArrayData> clazz, final CallSiteDescriptor desc, final LinkRequest request) { // array, index, value + final MethodType callType = desc.getMethodType(); + final Class<?> indexType = callType.parameterType(1); + final Class<?> elementType = callType.parameterType(2); + + if (ContinuousArrayData.class.isAssignableFrom(clazz) && indexType == int.class) { + final Object[] args = request.getArguments(); + final int index = (int)args[args.length - 2]; + + if (hasRoomFor(index)) { + MethodHandle setElement = getElementSetter(elementType); //Z(continuousarraydata, int, int), return true if successful + if (setElement != null) { + //else we are dealing with a wider type than supported by this callsite + MethodHandle getArray = ScriptObject.GET_ARRAY.methodHandle(); + getArray = MH.asType(getArray, getArray.type().changeReturnType(getClass())); + setElement = MH.filterArguments(setElement, 0, getArray); + final MethodHandle guard = MH.insertArguments(FAST_ACCESS_GUARD, 0, clazz); + return new GuardedInvocation(setElement, guard, (SwitchPoint)null, ClassCastException.class); //CCE if not a scriptObject anymore + } + } + } + + return null; + } + + /** + * Specialization - fast push implementation + * @param arg argument + * @return new array length + */ + public long fastPush(final int arg) { + throw new ClassCastException(String.valueOf(getClass())); //type is wrong, relink + } + + /** + * Specialization - fast push implementation + * @param arg argument + * @return new array length + */ + public long fastPush(final long arg) { + throw new ClassCastException(String.valueOf(getClass())); //type is wrong, relink + } + + /** + * Specialization - fast push implementation + * @param arg argument + * @return new array length + */ + public long fastPush(final double arg) { + throw new ClassCastException(String.valueOf(getClass())); //type is wrong, relink + } + + /** + * Specialization - fast push implementation + * @param arg argument + * @return new array length + */ + public long fastPush(final Object arg) { + throw new ClassCastException(String.valueOf(getClass())); //type is wrong, relink + } + + /** + * Specialization - fast pop implementation + * @return element value + */ + public int fastPopInt() { + throw new ClassCastException(String.valueOf(getClass())); //type is wrong, relink + } + + /** + * Specialization - fast pop implementation + * @return element value + */ + public long fastPopLong() { + throw new ClassCastException(String.valueOf(getClass())); //type is wrong, relink + } + + /** + * Specialization - fast pop implementation + * @return element value + */ + public double fastPopDouble() { + throw new ClassCastException(String.valueOf(getClass())); //type is wrong, relink + } + + /** + * Specialization - fast pop implementation + * @return element value + */ + public Object fastPopObject() { + throw new ClassCastException(String.valueOf(getClass())); //type is wrong, relink + } + + /** + * Specialization - fast concat implementation + * @param otherData data to concat + * @return new arraydata + */ + public ContinuousArrayData fastConcat(final ContinuousArrayData otherData) { + throw new ClassCastException(String.valueOf(getClass()) + " != " + String.valueOf(otherData.getClass())); + } +} |