blob: a92a0cccad8d256c61991d1aba927b42de6399b5 [file] [log] [blame]
/*
* 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 java.io.NotSerializableException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import jdk.nashorn.internal.codegen.types.Type;
/**
* This exception is thrown from an optimistic operation, e.g. an integer add,
* that was to optimistic for what really took place. Typically things like
* trying to get an array element that we want to be an int, and it was a double,
* and an int add that actually overflows and needs a double for the representation
*/
@SuppressWarnings("serial")
public final class UnwarrantedOptimismException extends RuntimeException {
/** Denotes an invalid program point */
public static final int INVALID_PROGRAM_POINT = -1;
/** The value for the first ordinary program point */
public static final int FIRST_PROGRAM_POINT = 1;
private Object returnValue;
private final int programPoint;
private final Type returnType;
/**
* Constructor without explicit return type. The return type is determined statically from the class of
* the return value, and only canonical internal number representations are recognized. Use
* {@link #createNarrowest} if you want to handle float and long values as numbers instead of objects.
*
* @param returnValue actual return value from the too narrow operation
* @param programPoint program point where unwarranted optimism was detected
*/
public UnwarrantedOptimismException(final Object returnValue, final int programPoint) {
this(returnValue, programPoint, getReturnType(returnValue));
}
/**
* Check if a program point is valid.
* @param programPoint the program point
* @return true if valid
*/
public static boolean isValid(final int programPoint) {
assert programPoint >= INVALID_PROGRAM_POINT;
return programPoint != INVALID_PROGRAM_POINT;
}
private static Type getReturnType(final Object v) {
if (v instanceof Double) {
return Type.NUMBER;
}
assert !(v instanceof Integer) : v + " is an int"; // Can't have an unwarranted optimism exception with int
return Type.OBJECT;
}
/**
* Constructor with explicit return value type.
* @param returnValue actual return value from the too narrow operation
* @param programPoint program point where unwarranted optimism was detected
* @param returnType type of the returned value. Used to disambiguate the return type. E.g. an {@code ObjectArrayData}
* might return a {@link Double} for a particular element getter, but still throw this exception even if the call
* site can accept a double, since the array's type is actually {@code Type#OBJECT}. In this case, it must
* explicitly use this constructor to indicate its values are to be considered {@code Type#OBJECT} and not
* {@code Type#NUMBER}.
*/
public UnwarrantedOptimismException(final Object returnValue, final int programPoint, final Type returnType) {
super("", null, false, Context.DEBUG);
assert returnType != Type.OBJECT || returnValue == null || !Type.typeFor(returnValue.getClass()).isNumeric();
assert returnType != Type.INT;
this.returnValue = returnValue;
this.programPoint = programPoint;
this.returnType = returnType;
}
/**
* Create an {@code UnwarrantedOptimismException} with the given return value and program point, narrowing
* the type to {@code number} if the value is a float or a long that can be represented as double.
*
* @param returnValue the return value
* @param programPoint the program point
* @return the exception
*/
public static UnwarrantedOptimismException createNarrowest(final Object returnValue, final int programPoint) {
if (returnValue instanceof Float
|| (returnValue instanceof Long && JSType.isRepresentableAsDouble((Long) returnValue))) {
return new UnwarrantedOptimismException(((Number) returnValue).doubleValue(), programPoint, Type.NUMBER);
}
return new UnwarrantedOptimismException(returnValue, programPoint);
}
/**
* Get the return value. This is a destructive readout, after the method is invoked the return value is null'd out.
* @return return value
*/
public Object getReturnValueDestructive() {
final Object retval = returnValue;
returnValue = null;
return retval;
}
Object getReturnValueNonDestructive() {
return returnValue;
}
/**
* Get the return type
* @return return type
*/
public Type getReturnType() {
return returnType;
}
/**
* Does this exception refer to an invalid program point? This might be OK if
* we throw it, e.g. from a parameter guard
* @return true if invalid program point specified
*/
public boolean hasInvalidProgramPoint() {
return programPoint == INVALID_PROGRAM_POINT;
}
/**
* Get the program point
* @return the program point
*/
public int getProgramPoint() {
return programPoint;
}
/**
* Check if we ended up with a primitive return value (even though it may be
* too wide for what we tried to do, e.g. double instead of int)
* @return true if return value is primitive
*/
public boolean hasPrimitiveReturnValue() {
return returnValue instanceof Number || returnValue instanceof Boolean;
}
@Override
public String getMessage() {
return "UNWARRANTED OPTIMISM: [returnValue=" +
returnValue +
" (class=" +
(returnValue == null ? "null" : returnValue.getClass().getSimpleName()) +
(hasInvalidProgramPoint() ?
" <invalid program point>" :
(" @ program point #" + programPoint)) +
")]";
}
private void writeObject(final ObjectOutputStream out) throws NotSerializableException {
throw new NotSerializableException(getClass().getName());
}
private void readObject(final ObjectInputStream in) throws NotSerializableException {
throw new NotSerializableException(getClass().getName());
}
}