diff --git a/src/hotspot/share/jvmci/jvmciCompilerToVM.cpp b/src/hotspot/share/jvmci/jvmciCompilerToVM.cpp
index 06961ab8f7a..b4c144c35e8 100644
--- a/src/hotspot/share/jvmci/jvmciCompilerToVM.cpp
+++ b/src/hotspot/share/jvmci/jvmciCompilerToVM.cpp
@@ -568,6 +568,33 @@ C2V_VMENTRY_NULL(jobject, lookupType, (JNIEnv* env, jobject, jstring jname, jcla
return JVMCIENV->get_jobject(result);
C2V_END
+C2V_VMENTRY_NULL(jobject, getArrayType, (JNIEnv* env, jobject, jobject jvmci_type))
+ if (jvmci_type == NULL) {
+ JVMCI_THROW_0(NullPointerException);
+ }
+
+ JVMCIObject jvmci_type_object = JVMCIENV->wrap(jvmci_type);
+ JVMCIKlassHandle array_klass(THREAD);
+ if (JVMCIENV->isa_HotSpotResolvedPrimitiveType(jvmci_type_object)) {
+ BasicType type = JVMCIENV->kindToBasicType(JVMCIENV->get_HotSpotResolvedPrimitiveType_kind(jvmci_type_object), JVMCI_CHECK_0);
+ if (type == T_VOID) {
+ return NULL;
+ }
+ array_klass = Universe::typeArrayKlassObj(type);
+ if (array_klass == NULL) {
+ JVMCI_THROW_MSG_NULL(InternalError, err_msg("No array klass for primitive type %s", type2name(type)));
+ }
+ } else {
+ Klass* klass = JVMCIENV->asKlass(jvmci_type);
+ if (klass == NULL) {
+ JVMCI_THROW_0(NullPointerException);
+ }
+ array_klass = klass->array_klass(CHECK_NULL);
+ }
+ JVMCIObject result = JVMCIENV->get_jvmci_type(array_klass, JVMCI_CHECK_NULL);
+ return JVMCIENV->get_jobject(result);
+C2V_END
+
C2V_VMENTRY_NULL(jobject, lookupClass, (JNIEnv* env, jobject, jclass mirror))
requireInHotSpot("lookupClass", JVMCI_CHECK_NULL);
if (mirror == NULL) {
@@ -2578,6 +2605,18 @@ C2V_VMENTRY_0(jboolean, addFailedSpeculation, (JNIEnv* env, jobject, jlong faile
return FailedSpeculation::add_failed_speculation(NULL, (FailedSpeculation**)(address) failed_speculations_address, (address) speculation, speculation_len);
}
+C2V_VMENTRY(void, callSystemExit, (JNIEnv* env, jobject, jint status))
+ JavaValue result(T_VOID);
+ JavaCallArguments jargs(1);
+ jargs.push_int(status);
+ JavaCalls::call_static(&result,
+ SystemDictionary::System_klass(),
+ vmSymbols::exit_method_name(),
+ vmSymbols::int_void_signature(),
+ &jargs,
+ CHECK);
+}
+
#define CC (char*) /*cast a literal from (const char*)*/
#define FN_PTR(f) CAST_FROM_FN_PTR(void*, &(c2v_ ## f))
@@ -2624,6 +2663,7 @@ JNINativeMethod CompilerToVM::methods[] = {
{CC "hasNeverInlineDirective", CC "(" HS_RESOLVED_METHOD ")Z", FN_PTR(hasNeverInlineDirective)},
{CC "shouldInlineMethod", CC "(" HS_RESOLVED_METHOD ")Z", FN_PTR(shouldInlineMethod)},
{CC "lookupType", CC "(" STRING HS_RESOLVED_KLASS "Z)" HS_RESOLVED_TYPE, FN_PTR(lookupType)},
+ {CC "getArrayType", CC "(" HS_RESOLVED_TYPE ")" HS_RESOLVED_KLASS, FN_PTR(getArrayType)},
{CC "lookupClass", CC "(" CLASS ")" HS_RESOLVED_TYPE, FN_PTR(lookupClass)},
{CC "lookupNameInPool", CC "(" HS_CONSTANT_POOL "I)" STRING, FN_PTR(lookupNameInPool)},
{CC "lookupNameAndTypeRefIndexInPool", CC "(" HS_CONSTANT_POOL "I)I", FN_PTR(lookupNameAndTypeRefIndexInPool)},
@@ -2723,6 +2763,7 @@ JNINativeMethod CompilerToVM::methods[] = {
{CC "getFailedSpeculationsAddress", CC "(" HS_RESOLVED_METHOD ")J", FN_PTR(getFailedSpeculationsAddress)},
{CC "releaseFailedSpeculations", CC "(J)V", FN_PTR(releaseFailedSpeculations)},
{CC "addFailedSpeculation", CC "(J[B)Z", FN_PTR(addFailedSpeculation)},
+ {CC "callSystemExit", CC "(I)V", FN_PTR(callSystemExit)},
};
int CompilerToVM::methods_count() {
diff --git a/src/hotspot/share/jvmci/jvmciEnv.cpp b/src/hotspot/share/jvmci/jvmciEnv.cpp
index 11a0281e4bb..a2b68b049b2 100644
--- a/src/hotspot/share/jvmci/jvmciEnv.cpp
+++ b/src/hotspot/share/jvmci/jvmciEnv.cpp
@@ -1361,6 +1361,9 @@ Handle JVMCIEnv::asConstant(JVMCIObject constant, JVMCI_TRAPS) {
return Handle(THREAD, obj);
} else if (isa_IndirectHotSpotObjectConstantImpl(constant)) {
jlong object_handle = get_IndirectHotSpotObjectConstantImpl_objectHandle(constant);
+ if (object_handle == 0L) {
+ JVMCI_THROW_MSG_(NullPointerException, "Foreign object reference has been cleared", Handle());
+ }
oop result = resolve_handle(object_handle);
if (result == NULL) {
JVMCI_THROW_MSG_(InternalError, "Constant was unexpectedly NULL", Handle());
diff --git a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/CompilerToVM.java b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/CompilerToVM.java
index 2e313f7d1ca..28dc88c38ad 100644
--- a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/CompilerToVM.java
+++ b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/CompilerToVM.java
@@ -770,6 +770,12 @@ final class CompilerToVM {
*/
native HotSpotResolvedJavaType getComponentType(HotSpotResolvedObjectTypeImpl type);
+ /**
+ * Get the array class for {@code type}. This can't be done symbolically since anonymous types
+ * can't be looked up by name.
+ */
+ native HotSpotResolvedObjectTypeImpl getArrayType(HotSpotResolvedJavaType type);
+
/**
* Forces initialization of {@code type}.
*/
@@ -978,4 +984,9 @@ final class CompilerToVM {
* @see HotSpotJVMCIRuntime#detachCurrentThread()
*/
native void detachCurrentThread();
+
+ /**
+ * @see HotSpotJVMCIRuntime#exitHotSpot(int)
+ */
+ native void callSystemExit(int status);
}
diff --git a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotCompilationRequest.java b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotCompilationRequest.java
index 332dd26e6d2..cfc3d86056a 100644
--- a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotCompilationRequest.java
+++ b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotCompilationRequest.java
@@ -87,4 +87,9 @@ public class HotSpotCompilationRequest extends CompilationRequest {
public int getId() {
return id;
}
+
+ @Override
+ public String toString() {
+ return id + ":" + super.toString();
+ }
}
diff --git a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotJVMCIRuntime.java b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotJVMCIRuntime.java
index 6cffec65634..f10055a28cd 100644
--- a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotJVMCIRuntime.java
+++ b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotJVMCIRuntime.java
@@ -223,6 +223,8 @@ public final class HotSpotJVMCIRuntime implements JVMCIRuntime {
// so that -XX:+JVMCIPrintProperties shows the option.
InitTimer(Boolean.class, false, "Specifies if initialization timing is enabled."),
PrintConfig(Boolean.class, false, "Prints VM configuration available via JVMCI."),
+ AuditHandles(Boolean.class, false, "Record stack trace along with scoped foreign object reference wrappers " +
+ "to debug issue with a wrapper being used after its scope has closed."),
TraceMethodDataFilter(String.class, null,
"Enables tracing of profiling info when read by JVMCI.",
"Empty value: trace all methods",
@@ -687,9 +689,11 @@ assert factories != null : "sanity";
return Collections.unmodifiableMap(backends);
}
+ @SuppressWarnings("try")
@VMEntryPoint
private HotSpotCompilationRequestResult compileMethod(HotSpotResolvedJavaMethod method, int entryBCI, long compileState, int id) {
- CompilationRequestResult result = getCompiler().compileMethod(new HotSpotCompilationRequest(method, entryBCI, compileState, id));
+ HotSpotCompilationRequest request = new HotSpotCompilationRequest(method, entryBCI, compileState, id);
+ CompilationRequestResult result = getCompiler().compileMethod(request);
assert result != null : "compileMethod must always return something";
HotSpotCompilationRequestResult hsResult;
if (result instanceof HotSpotCompilationRequestResult) {
@@ -704,7 +708,6 @@ assert factories != null : "sanity";
hsResult = HotSpotCompilationRequestResult.success(inlinedBytecodes);
}
}
-
return hsResult;
}
@@ -1032,4 +1035,14 @@ assert factories != null : "sanity";
public void excludeFromJVMCICompilation(Module...modules) {
this.excludeFromJVMCICompilation = modules.clone();
}
+
+ /**
+ * Calls {@link System#exit(int)} in HotSpot's runtime.
+ */
+ public void exitHotSpot(int status) {
+ if (!IS_IN_NATIVE_IMAGE) {
+ System.exit(status);
+ }
+ compilerToVm.callSystemExit(status);
+ }
}
diff --git a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotObjectConstantScope.java b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotObjectConstantScope.java
new file mode 100644
index 00000000000..0ef506000fe
--- /dev/null
+++ b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotObjectConstantScope.java
@@ -0,0 +1,121 @@
+/*
+ * Copyright (c) 2019, 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.
+ */
+package jdk.vm.ci.hotspot;
+
+import java.util.ArrayList;
+import java.util.List;
+import java.util.Objects;
+
+import jdk.vm.ci.services.Services;
+
+/**
+ * A mechanism for limiting the lifetime of a foreign object reference encapsulated in a
+ * {@link HotSpotObjectConstant}.
+ *
+ * A {@link HotSpotObjectConstant} allocated in a {@linkplain #openLocalScope local} scope will have
+ * its reference to any foreign object cleared when the scope {@linkplain #close() closes}. This
+ * allows the foreign memory manager to reclaim the foreign object (once there are no other strong
+ * references to it).
+ *
+ * {@link HotSpotObjectConstantScope}s have no impact on {@link HotSpotObjectConstant}s that do not
+ * encapsulate a foreign object reference.
+ *
+ * The object returned by {@link #enterGlobalScope()} or {@link #openLocalScope(Object)} should
+ * always be used in a try-with-resources statement. Failure to close a scope will almost certainly
+ * result in foreign objects being leaked.
+ */
+public final class HotSpotObjectConstantScope implements AutoCloseable {
+ static final ThreadLocal CURRENT = new ThreadLocal<>();
+
+ private final HotSpotObjectConstantScope parent;
+ private List foreignObjects;
+
+ /**
+ * An object whose {@link Object#toString()} value describes a non-global scope. This is
+ * {@code null} iff this is a global scope.
+ */
+ final Object localScopeDescription;
+
+ /**
+ * Opens a local scope that upon closing, will release foreign object references encapsulated by
+ * {@link HotSpotObjectConstant}s created in the scope.
+ *
+ * @param description an non-null object whose {@link Object#toString()} value describes the
+ * scope being opened
+ * @return {@code null} if the current runtime does not support remote object references
+ */
+ public static HotSpotObjectConstantScope openLocalScope(Object description) {
+ return Services.IS_IN_NATIVE_IMAGE ? new HotSpotObjectConstantScope(Objects.requireNonNull(description)) : null;
+ }
+
+ /**
+ * Enters the global scope. This is useful to escape a local scope for execution that will
+ * create foreign object references that need to outlive the local scope.
+ *
+ * Foreign object references encapsulated by {@link HotSpotObjectConstant}s created in the
+ * global scope are only subject to reclamation once the {@link HotSpotObjectConstant} wrapper
+ * dies.
+ *
+ * @return {@code null} if the current runtime does not support remote object references or if
+ * this thread is currently in the global scope
+ */
+ public static HotSpotObjectConstantScope enterGlobalScope() {
+ return Services.IS_IN_NATIVE_IMAGE && CURRENT.get() != null ? new HotSpotObjectConstantScope(null) : null;
+ }
+
+ private HotSpotObjectConstantScope(Object localScopeDescription) {
+ this.parent = CURRENT.get();
+ CURRENT.set(this);
+ this.localScopeDescription = localScopeDescription;
+ }
+
+ /**
+ * Determines if this scope is global.
+ */
+ boolean isGlobal() {
+ return localScopeDescription == null;
+ }
+
+ void add(IndirectHotSpotObjectConstantImpl obj) {
+ assert !isGlobal();
+ if (foreignObjects == null) {
+ foreignObjects = new ArrayList<>();
+ }
+ foreignObjects.add(obj);
+ }
+
+ @VMEntryPoint
+ @Override
+ public void close() {
+ if (CURRENT.get() != this) {
+ throw new IllegalStateException("Cannot close non-active scope");
+ }
+ if (foreignObjects != null) {
+ for (IndirectHotSpotObjectConstantImpl obj : foreignObjects) {
+ obj.clear(localScopeDescription);
+ }
+ foreignObjects = null;
+ }
+ CURRENT.set(parent);
+ }
+}
diff --git a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotResolvedJavaType.java b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotResolvedJavaType.java
index e1949c0478f..0113db3b9b6 100644
--- a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotResolvedJavaType.java
+++ b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotResolvedJavaType.java
@@ -22,11 +22,15 @@
*/
package jdk.vm.ci.hotspot;
+import static jdk.vm.ci.hotspot.HotSpotJVMCIRuntime.runtime;
+
import jdk.vm.ci.meta.JavaConstant;
import jdk.vm.ci.meta.ResolvedJavaType;
public abstract class HotSpotResolvedJavaType extends HotSpotJavaType implements ResolvedJavaType {
+ HotSpotResolvedObjectTypeImpl arrayOfType;
+
HotSpotResolvedJavaType(String name) {
super(name);
}
@@ -40,4 +44,12 @@ public abstract class HotSpotResolvedJavaType extends HotSpotJavaType implements
}
abstract JavaConstant getJavaMirror();
+
+ @Override
+ public HotSpotResolvedObjectType getArrayClass() {
+ if (arrayOfType == null) {
+ arrayOfType = runtime().compilerToVm.getArrayType(this);
+ }
+ return arrayOfType;
+ }
}
diff --git a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotResolvedObjectTypeImpl.java b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotResolvedObjectTypeImpl.java
index c17aaf493f7..56579eefacc 100644
--- a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotResolvedObjectTypeImpl.java
+++ b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotResolvedObjectTypeImpl.java
@@ -72,7 +72,6 @@ final class HotSpotResolvedObjectTypeImpl extends HotSpotResolvedJavaType implem
private volatile HotSpotResolvedJavaField[] instanceFields;
private volatile HotSpotResolvedObjectTypeImpl[] interfaces;
private HotSpotConstantPool constantPool;
- private HotSpotResolvedObjectType arrayOfType;
private final JavaConstant mirror;
private HotSpotResolvedObjectTypeImpl superClass;
@@ -103,17 +102,24 @@ final class HotSpotResolvedObjectTypeImpl extends HotSpotResolvedJavaType implem
* Creates the JVMCI mirror for a {@link Class} object.
*
* NOTE: Creating an instance of this class does not install the mirror for the
- * {@link Class} type. {@link #fromMetaspace} instead.
+ * {@link Class} type.
*
*
* @param metadataPointer the Klass* to create the mirror for
*/
+ @SuppressWarnings("try")
HotSpotResolvedObjectTypeImpl(long metadataPointer, String name) {
super(name);
- this.metadataPointer = metadataPointer;
- this.mirror = runtime().compilerToVm.getJavaMirror(this);
assert metadataPointer != 0;
- assert getName().charAt(0) != '[' || isArray() : getName();
+ this.metadataPointer = metadataPointer;
+
+ // The mirror object must be in the global scope since
+ // this object will be cached in HotSpotJVMCIRuntime.resolvedJavaTypes
+ // and live across more than one compilation.
+ try (HotSpotObjectConstantScope global = HotSpotObjectConstantScope.enterGlobalScope()) {
+ this.mirror = runtime().compilerToVm.getJavaMirror(this);
+ assert getName().charAt(0) != '[' || isArray() : getName();
+ }
}
/**
@@ -146,18 +152,6 @@ final class HotSpotResolvedObjectTypeImpl extends HotSpotResolvedJavaType implem
return UNSAFE.getInt(getMetaspaceKlass() + config.klassAccessFlagsOffset);
}
- @Override
- public HotSpotResolvedObjectType getArrayClass() {
- if (arrayOfType == null) {
- try {
- arrayOfType = (HotSpotResolvedObjectType) runtime().compilerToVm.lookupType("[" + getName(), this, true);
- } catch (ClassNotFoundException e) {
- throw new JVMCIError(e);
- }
- }
- return arrayOfType;
- }
-
@Override
public ResolvedJavaType getComponentType() {
return runtime().compilerToVm.getComponentType(this);
@@ -580,6 +574,11 @@ final class HotSpotResolvedObjectTypeImpl extends HotSpotResolvedJavaType implem
return null;
}
+ if (resolvedMethod.canBeStaticallyBound()) {
+ // No assumptions are required.
+ return new AssumptionResult<>(resolvedMethod);
+ }
+
ResolvedJavaMethod result = resolvedMethod.uniqueConcreteMethod(this);
if (result != null) {
return new AssumptionResult<>(result, new ConcreteMethod(method, this, result));
diff --git a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotResolvedPrimitiveType.java b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotResolvedPrimitiveType.java
index 194e0c1d298..02508caa91f 100644
--- a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotResolvedPrimitiveType.java
+++ b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotResolvedPrimitiveType.java
@@ -46,7 +46,6 @@ public final class HotSpotResolvedPrimitiveType extends HotSpotResolvedJavaType
@NativeImageReinitialize static HotSpotResolvedPrimitiveType[] primitives;
private JavaKind kind;
- private HotSpotResolvedObjectType arrayClass;
HotSpotObjectConstantImpl mirror;
/**
@@ -87,14 +86,7 @@ public final class HotSpotResolvedPrimitiveType extends HotSpotResolvedJavaType
if (kind == JavaKind.Void) {
return null;
}
- if (arrayClass == null) {
- try {
- arrayClass = (HotSpotResolvedObjectType) runtime().compilerToVm.lookupType("[" + kind.getTypeChar(), null, true);
- } catch (ClassNotFoundException e) {
- throw new JVMCIError(e);
- }
- }
- return arrayClass;
+ return super.getArrayClass();
}
@Override
diff --git a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/IndirectHotSpotObjectConstantImpl.java b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/IndirectHotSpotObjectConstantImpl.java
index 1a1dbd1a544..3d65879fc41 100644
--- a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/IndirectHotSpotObjectConstantImpl.java
+++ b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/IndirectHotSpotObjectConstantImpl.java
@@ -24,17 +24,52 @@ package jdk.vm.ci.hotspot;
import static jdk.vm.ci.hotspot.HotSpotJVMCIRuntime.runtime;
+import java.io.ByteArrayOutputStream;
+import java.io.PrintStream;
+
+import jdk.vm.ci.hotspot.HotSpotJVMCIRuntime.Option;
import jdk.vm.ci.meta.JavaConstant;
+/**
+ * Encapsulates a JNI reference to an object in the HotSpot heap.
+ *
+ * {@link IndirectHotSpotObjectConstantImpl} objects are only allocated in the shared library heap.
+ *
+ * @see HotSpotObjectConstantScope
+ */
final class IndirectHotSpotObjectConstantImpl extends HotSpotObjectConstantImpl {
/**
- * An object handle in {@code JVMCI::_jvmci_handles}.
+ * An object handle in {@code JVMCI::_object_handles}.
+ */
+ private long objectHandle;
+
+ /**
+ * Lazily computed hash code.
*/
- final long objectHandle;
private int hashCode;
final IndirectHotSpotObjectConstantImpl base;
+ private static class Audit {
+ final Object scope;
+ final long handle;
+ final Throwable origin;
+
+ Audit(Object scope, long handle, Throwable origin) {
+ this.scope = scope;
+ this.handle = handle;
+ this.origin = origin;
+ }
+ }
+
+ /**
+ * Details useful to audit a scoped handle used after its creating scope closes. Set to an
+ * {@link Audit} object if {@link HotSpotJVMCIRuntime.Option#AuditHandles} is true otherwise to
+ * {@link HotSpotObjectConstantScope#localScopeDescription}.
+ */
+ private Object rawAudit;
+
+ @SuppressWarnings("serial")
@VMEntryPoint
private IndirectHotSpotObjectConstantImpl(long objectHandle, boolean compressed, boolean skipRegister) {
super(compressed);
@@ -42,7 +77,20 @@ final class IndirectHotSpotObjectConstantImpl extends HotSpotObjectConstantImpl
this.objectHandle = objectHandle;
this.base = null;
if (!skipRegister) {
- HandleCleaner.create(this, objectHandle);
+ HotSpotObjectConstantScope scope = HotSpotObjectConstantScope.CURRENT.get();
+ if (scope != null && !scope.isGlobal()) {
+ scope.add(this);
+ if (HotSpotJVMCIRuntime.Option.AuditHandles.getBoolean()) {
+ rawAudit = new Audit(scope.localScopeDescription, objectHandle, new Throwable() {
+ @Override
+ public String toString() {
+ return "Created " + objectHandle;
+ }
+ });
+ }
+ } else {
+ HandleCleaner.create(this, objectHandle);
+ }
}
}
@@ -50,12 +98,59 @@ final class IndirectHotSpotObjectConstantImpl extends HotSpotObjectConstantImpl
super(compressed);
// This is a variant of an original object that only varies in compress vs uncompressed.
// Instead of creating a new handle, reference that object and objectHandle.
- this.objectHandle = base.objectHandle;
- // There should only be on level of indirection to the base object.
+ this.objectHandle = base.getHandle();
+ // There should only be one level of indirection to the base object.
assert base.base == null || base.base.base == null;
this.base = base.base != null ? base.base : base;
}
+ long getHandle() {
+ checkHandle();
+ return objectHandle;
+ }
+
+ private void checkHandle() {
+ if (objectHandle == 0L) {
+ String message;
+ if (rawAudit instanceof Audit) {
+ Audit audit = (Audit) rawAudit;
+ ByteArrayOutputStream baos = new ByteArrayOutputStream();
+ PrintStream ps = new PrintStream(baos);
+ ps.println("Foreign object reference " + audit.handle + " created in scope '" + audit.scope + "' is no longer valid. Origin: {");
+ audit.origin.printStackTrace(ps);
+ ps.print('}');
+ ps.flush();
+ message = baos.toString();
+ } else {
+ message = "Foreign object reference created in scope '" + rawAudit + "' is no longer valid. " +
+ "Set property " + Option.AuditHandles.getPropertyName() + "=true to show origin of invalid foreign references.";
+ }
+ throw new NullPointerException(message);
+ }
+ }
+
+ boolean isValid() {
+ return objectHandle != 0L;
+ }
+
+ @Override
+ public HotSpotResolvedObjectType getType() {
+ checkHandle();
+ return super.getType();
+ }
+
+ /**
+ * Clears the foreign object reference.
+ */
+ void clear(Object scopeDescription) {
+ checkHandle();
+ CompilerToVM.compilerToVM().deleteGlobalHandle(objectHandle);
+ if (rawAudit == null) {
+ rawAudit = scopeDescription;
+ }
+ objectHandle = 0L;
+ }
+
@Override
public JavaConstant compress() {
assert !compressed;
@@ -70,6 +165,7 @@ final class IndirectHotSpotObjectConstantImpl extends HotSpotObjectConstantImpl
@Override
public int getIdentityHashCode() {
+ checkHandle();
int hash = hashCode;
if (hash == 0) {
hash = runtime().compilerToVm.getIdentityHashCode(this);
diff --git a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/SharedLibraryJVMCIReflection.java b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/SharedLibraryJVMCIReflection.java
index 684c4e00564..12138a70d35 100644
--- a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/SharedLibraryJVMCIReflection.java
+++ b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/SharedLibraryJVMCIReflection.java
@@ -114,7 +114,7 @@ class SharedLibraryJVMCIReflection extends HotSpotJVMCIReflection {
}
IndirectHotSpotObjectConstantImpl indirectX = (IndirectHotSpotObjectConstantImpl) x;
IndirectHotSpotObjectConstantImpl indirectY = (IndirectHotSpotObjectConstantImpl) y;
- return runtime().compilerToVm.equals(x, indirectX.objectHandle, y, indirectY.objectHandle);
+ return runtime().compilerToVm.equals(x, indirectX.getHandle(), y, indirectY.getHandle());
}
@Override
@@ -288,6 +288,10 @@ class SharedLibraryJVMCIReflection extends HotSpotJVMCIReflection {
DirectHotSpotObjectConstantImpl direct = (DirectHotSpotObjectConstantImpl) object;
return "CompilerObject<" + direct.object.getClass().getName() + ">";
}
+ IndirectHotSpotObjectConstantImpl indirect = (IndirectHotSpotObjectConstantImpl) object;
+ if (!indirect.isValid()) {
+ return "Instance";
+ }
return "Instance<" + object.getType().toJavaName() + ">";
}
diff --git a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.meta/src/jdk/vm/ci/meta/MetaUtil.java b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.meta/src/jdk/vm/ci/meta/MetaUtil.java
index 257efa981a2..528bb34e059 100644
--- a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.meta/src/jdk/vm/ci/meta/MetaUtil.java
+++ b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.meta/src/jdk/vm/ci/meta/MetaUtil.java
@@ -86,6 +86,29 @@ public class MetaUtil {
}
}
+ /**
+ * Classes for lambdas can have {@code /} characters that are not package separators. These are
+ * distinguished by being followed by a character that is not a
+ * {@link Character#isJavaIdentifierStart(char)} (e.g.,
+ * "jdk.vm.ci.runtime.test.TypeUniverse$$Lambda$1/869601985").
+ */
+ private static String replacePackageSeparatorsWithDot(String name) {
+ int length = name.length();
+ int i = 0;
+ StringBuilder buf = new StringBuilder(length);
+ while (i < length - 1) {
+ char ch = name.charAt(i);
+ if (ch == '/' && Character.isJavaIdentifierStart(name.charAt(i + 1))) {
+ buf.append('.');
+ } else {
+ buf.append(ch);
+ }
+ i++;
+ }
+ buf.append(name.charAt(length - 1));
+ return buf.toString();
+ }
+
/**
* Converts a type name in internal form to an external form.
*
@@ -99,7 +122,7 @@ public class MetaUtil {
public static String internalNameToJava(String name, boolean qualified, boolean classForNameCompatible) {
switch (name.charAt(0)) {
case 'L': {
- String result = name.substring(1, name.length() - 1).replace('/', '.');
+ String result = replacePackageSeparatorsWithDot(name.substring(1, name.length() - 1));
if (!qualified) {
final int lastDot = result.lastIndexOf('.');
if (lastDot != -1) {
@@ -109,7 +132,7 @@ public class MetaUtil {
return result;
}
case '[':
- return classForNameCompatible ? name.replace('/', '.') : internalNameToJava(name.substring(1), qualified, classForNameCompatible) + "[]";
+ return classForNameCompatible ? replacePackageSeparatorsWithDot(name) : internalNameToJava(name.substring(1), qualified, classForNameCompatible) + "[]";
default:
if (name.length() != 1) {
throw new IllegalArgumentException("Illegal internal name: " + name);
diff --git a/test/hotspot/jtreg/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestMetaAccessProvider.java b/test/hotspot/jtreg/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestMetaAccessProvider.java
index ce5f47d3cb1..33ff2510df0 100644
--- a/test/hotspot/jtreg/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestMetaAccessProvider.java
+++ b/test/hotspot/jtreg/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestMetaAccessProvider.java
@@ -68,16 +68,22 @@ public class TestMetaAccessProvider extends TypeUniverse {
metaAccess.encodeDeoptActionAndReason(DEOPT_ACTION, DEOPT_REASON, DEBUG_IDS[3]).asInt()
};
+ private static boolean isUnsafeAnoymous(ResolvedJavaType type) {
+ return type.getHostClass() != null;
+ }
+
@Test
public void lookupJavaTypeTest() {
for (Class> c : classes) {
ResolvedJavaType type = metaAccess.lookupJavaType(c);
assertNotNull(c.toString(), type);
- assertEquals(c.toString(), type.getName(), toInternalName(c.getName()));
- assertEquals(c.toString(), type.getName(), toInternalName(type.toJavaName()));
- assertEquals(c.toString(), c.getName(), type.toClassName());
- if (!type.isArray()) {
- assertEquals(c.toString(), c.getName(), type.toJavaName());
+ if (!isUnsafeAnoymous(type)) {
+ assertEquals(c.toString(), type.getName(), toInternalName(c.getName()));
+ assertEquals(c.toString(), type.getName(), toInternalName(type.toJavaName()));
+ assertEquals(c.toString(), c.getName(), type.toClassName());
+ if (!type.isArray()) {
+ assertEquals(c.toString(), c.getName(), type.toJavaName());
+ }
}
}
}
@@ -92,7 +98,9 @@ public class TestMetaAccessProvider extends TypeUniverse {
ResolvedJavaType[] result = metaAccess.lookupJavaTypes(classes.toArray(new Class>[classes.size()]));
int counter = 0;
for (Class> aClass : classes) {
- assertEquals("Unexpected javaType: " + result[counter] + " while expecting of class: " + aClass, result[counter].toClassName(), aClass.getName());
+ if (!isUnsafeAnoymous(result[counter])) {
+ assertEquals("Unexpected javaType: " + result[counter] + " while expecting of class: " + aClass, result[counter].toClassName(), aClass.getName());
+ }
counter++;
}
}
diff --git a/test/hotspot/jtreg/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestResolvedJavaType.java b/test/hotspot/jtreg/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestResolvedJavaType.java
index fe2cbe069eb..349b05948c5 100644
--- a/test/hotspot/jtreg/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestResolvedJavaType.java
+++ b/test/hotspot/jtreg/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestResolvedJavaType.java
@@ -161,7 +161,11 @@ public class TestResolvedJavaType extends TypeUniverse {
for (Class> c : classes) {
ResolvedJavaType type = metaAccess.lookupJavaType(c);
ResolvedJavaType host = type.getHostClass();
- assertNull(host);
+ if (!type.equals(predicateType)) {
+ assertNull(host);
+ } else {
+ assertNotNull(host);
+ }
}
class LocalClass {}
diff --git a/test/hotspot/jtreg/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TypeUniverse.java b/test/hotspot/jtreg/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TypeUniverse.java
index 87a16d20ce8..1416c317d3e 100644
--- a/test/hotspot/jtreg/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TypeUniverse.java
+++ b/test/hotspot/jtreg/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TypeUniverse.java
@@ -51,6 +51,7 @@ import java.util.Map;
import java.util.Queue;
import java.util.Set;
import java.util.TreeMap;
+import java.util.function.Predicate;
import java.util.stream.Collectors;
import static java.lang.reflect.Modifier.isFinal;
@@ -68,6 +69,7 @@ public class TypeUniverse {
public static final ConstantReflectionProvider constantReflection = JVMCI.getRuntime().getHostJVMCIBackend().getConstantReflection();
public static final Collection> classes = new HashSet<>();
public static final Set javaTypes;
+ public static final ResolvedJavaType predicateType;
public static final Map, Class>> arrayClasses = new HashMap<>();
private static List constants;
@@ -116,6 +118,9 @@ public class TypeUniverse {
for (Class> c : initialClasses) {
addClass(c);
}
+ Predicate predicate = s -> s.length() == 1;
+ addClass(predicate.getClass());
+ predicateType = metaAccess.lookupJavaType(predicate.getClass());
javaTypes = Collections.unmodifiableSet(classes.stream().map(c -> metaAccess.lookupJavaType(c)).collect(Collectors.toSet()));
}