Skip to content

Commit 2993654

Browse files
author
Tom Rodriguez
committedJun 24, 2020
8247246: Add explicit ResolvedJavaType.link and expose presence of default methods
Reviewed-by: kvn
1 parent 6715f23 commit 2993654

File tree

9 files changed

+206
-2
lines changed

9 files changed

+206
-2
lines changed
 

‎src/hotspot/share/jvmci/jvmciCompilerToVM.cpp

+13
Original file line numberDiff line numberDiff line change
@@ -1768,6 +1768,18 @@ C2V_VMENTRY(void, ensureInitialized, (JNIEnv* env, jobject, jobject jvmci_type))
17681768
}
17691769
C2V_END
17701770

1771+
C2V_VMENTRY(void, ensureLinked, (JNIEnv* env, jobject, jobject jvmci_type))
1772+
if (jvmci_type == NULL) {
1773+
JVMCI_THROW(NullPointerException);
1774+
}
1775+
1776+
Klass* klass = JVMCIENV->asKlass(jvmci_type);
1777+
if (klass != NULL && klass->is_instance_klass()) {
1778+
InstanceKlass* k = InstanceKlass::cast(klass);
1779+
k->link_class(CHECK);
1780+
}
1781+
C2V_END
1782+
17711783
C2V_VMENTRY_0(jint, interpreterFrameSize, (JNIEnv* env, jobject, jobject bytecode_frame_handle))
17721784
if (bytecode_frame_handle == NULL) {
17731785
JVMCI_THROW_0(NullPointerException);
@@ -2775,6 +2787,7 @@ JNINativeMethod CompilerToVM::methods[] = {
27752787
{CC "getInterfaces", CC "(" HS_RESOLVED_KLASS ")[" HS_RESOLVED_KLASS, FN_PTR(getInterfaces)},
27762788
{CC "getComponentType", CC "(" HS_RESOLVED_KLASS ")" HS_RESOLVED_TYPE, FN_PTR(getComponentType)},
27772789
{CC "ensureInitialized", CC "(" HS_RESOLVED_KLASS ")V", FN_PTR(ensureInitialized)},
2790+
{CC "ensureLinked", CC "(" HS_RESOLVED_KLASS ")V", FN_PTR(ensureLinked)},
27782791
{CC "getIdentityHashCode", CC "(" OBJECTCONSTANT ")I", FN_PTR(getIdentityHashCode)},
27792792
{CC "isInternedString", CC "(" OBJECTCONSTANT ")Z", FN_PTR(isInternedString)},
27802793
{CC "unboxPrimitive", CC "(" OBJECTCONSTANT ")" OBJECT, FN_PTR(unboxPrimitive)},

‎src/hotspot/share/jvmci/vmStructs_jvmci.cpp

+7
Original file line numberDiff line numberDiff line change
@@ -554,7 +554,14 @@
554554
declare_constant(InstanceKlass::linked) \
555555
declare_constant(InstanceKlass::being_initialized) \
556556
declare_constant(InstanceKlass::fully_initialized) \
557+
\
558+
/*********************************/ \
559+
/* InstanceKlass _misc_flags */ \
560+
/*********************************/ \
561+
\
557562
declare_constant(InstanceKlass::_misc_is_unsafe_anonymous) \
563+
declare_constant(InstanceKlass::_misc_has_nonstatic_concrete_methods) \
564+
declare_constant(InstanceKlass::_misc_declares_nonstatic_concrete_methods) \
558565
\
559566
declare_constant(JumpData::taken_off_set) \
560567
declare_constant(JumpData::displacement_off_set) \

‎src/jdk.hotspot.agent/macosx/native/libsaproc/libproc_impl.c

+6
Original file line numberDiff line numberDiff line change
@@ -56,6 +56,8 @@ int pathmap_open(const char* name) {
5656
if (fd >= 0) {
5757
print_debug("path %s substituted for %s\n", alt_path, name);
5858
return fd;
59+
} else {
60+
print_debug("can't open %s\n", alt_path);
5961
}
6062

6163
if (strrchr(name, '/')) {
@@ -65,12 +67,16 @@ int pathmap_open(const char* name) {
6567
if (fd >= 0) {
6668
print_debug("path %s substituted for %s\n", alt_path, name);
6769
return fd;
70+
} else {
71+
print_debug("can't open %s\n", alt_path);
6872
}
6973
}
7074
} else {
7175
fd = open(name, O_RDONLY);
7276
if (fd >= 0) {
7377
return fd;
78+
} else {
79+
print_debug("can't open %s\n", name);
7480
}
7581
}
7682
return -1;

‎src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/CompilerToVM.java

+5
Original file line numberDiff line numberDiff line change
@@ -772,6 +772,11 @@ HotSpotResolvedObjectTypeImpl getResolvedJavaType(long displacement, boolean com
772772
*/
773773
native void ensureInitialized(HotSpotResolvedObjectTypeImpl type);
774774

775+
/**
776+
* Forces linking of {@code type}.
777+
*/
778+
native void ensureLinked(HotSpotResolvedObjectTypeImpl type);
779+
775780
/**
776781
* Checks if {@code object} is a String and is an interned string value.
777782
*/

‎src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotResolvedObjectTypeImpl.java

+21
Original file line numberDiff line numberDiff line change
@@ -372,6 +372,27 @@ public boolean isLinked() {
372372
return isArray() ? true : getInitState() >= config().instanceKlassStateLinked;
373373
}
374374

375+
@Override
376+
public void link() {
377+
if (!isLinked()) {
378+
runtime().compilerToVm.ensureLinked(this);
379+
}
380+
}
381+
382+
@Override
383+
public boolean hasDefaultMethods() {
384+
HotSpotVMConfig config = config();
385+
int miscFlags = UNSAFE.getChar(getMetaspaceKlass() + config.instanceKlassMiscFlagsOffset);
386+
return (miscFlags & config.jvmMiscFlagsHasDefaultMethods) != 0;
387+
}
388+
389+
@Override
390+
public boolean declaresDefaultMethods() {
391+
HotSpotVMConfig config = config();
392+
int miscFlags = UNSAFE.getChar(getMetaspaceKlass() + config.instanceKlassMiscFlagsOffset);
393+
return (miscFlags & config.jvmMiscFlagsDeclaresDefaultMethods) != 0;
394+
}
395+
375396
/**
376397
* Returns the value of the state field {@code InstanceKlass::_init_state} of the metaspace
377398
* klass.

‎src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotResolvedPrimitiveType.java

+14
Original file line numberDiff line numberDiff line change
@@ -250,6 +250,20 @@ public ResolvedJavaType resolve(ResolvedJavaType accessingClass) {
250250
public void initialize() {
251251
}
252252

253+
@Override
254+
public void link() {
255+
}
256+
257+
@Override
258+
public boolean hasDefaultMethods() {
259+
return false;
260+
}
261+
262+
@Override
263+
public boolean declaresDefaultMethods() {
264+
return false;
265+
}
266+
253267
@Override
254268
public ResolvedJavaField findInstanceFieldWithOffset(long offset, JavaKind expectedType) {
255269
return null;

‎src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotVMConfig.java

+3
Original file line numberDiff line numberDiff line change
@@ -137,6 +137,9 @@ String getHostArchitectureName() {
137137
final int jvmAccEnum = getConstant("JVM_ACC_ENUM", Integer.class);
138138
final int jvmAccInterface = getConstant("JVM_ACC_INTERFACE", Integer.class);
139139

140+
final int jvmMiscFlagsHasDefaultMethods = getConstant("InstanceKlass::_misc_has_nonstatic_concrete_methods", Integer.class);
141+
final int jvmMiscFlagsDeclaresDefaultMethods = getConstant("InstanceKlass::_misc_declares_nonstatic_concrete_methods", Integer.class);
142+
140143
// This is only valid on AMD64.
141144
final int runtimeCallStackSize = getConstant("frame::arg_reg_save_area_bytes", Integer.class, osArch.equals("amd64") ? null : 0);
142145

‎src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.meta/src/jdk/vm/ci/meta/ResolvedJavaType.java

+28-2
Original file line numberDiff line numberDiff line change
@@ -105,6 +105,30 @@ default boolean isLeaf() {
105105
*/
106106
boolean isLinked();
107107

108+
/**
109+
* Links this type. If this method returns normally, then future calls of {@link #isLinked} will
110+
* return true and future calls of {@link #link} are no-ops. If the method throws an exception,
111+
* then future calls of {@link #isLinked} will return false and future calls of {@link #link}
112+
* will reattempt the linking step which might succeed or throw an exception.
113+
*/
114+
default void link() {
115+
throw new UnsupportedOperationException("link is unsupported");
116+
}
117+
118+
/**
119+
* Checks whether this type or any of its supertypes or superinterfaces has default methods.
120+
*/
121+
default boolean hasDefaultMethods() {
122+
throw new UnsupportedOperationException("hasDefaultMethods is unsupported");
123+
}
124+
125+
/**
126+
* Checks whether this type declares defaults methods.
127+
*/
128+
default boolean declaresDefaultMethods() {
129+
throw new UnsupportedOperationException("declaresDefaultMethods is unsupported");
130+
}
131+
108132
/**
109133
* Determines if this type is either the same as, or is a superclass or superinterface of, the
110134
* type represented by the specified parameter. This method is identical to
@@ -308,13 +332,15 @@ default ResolvedJavaMethod resolveConcreteMethod(ResolvedJavaMethod method, Reso
308332

309333
/**
310334
* Returns an array reflecting all the constructors declared by this type. This method is
311-
* similar to {@link Class#getDeclaredConstructors()} in terms of returned constructors.
335+
* similar to {@link Class#getDeclaredConstructors()} in terms of returned constructors. Calling
336+
* this method forces this type to be {@link #link linked}.
312337
*/
313338
ResolvedJavaMethod[] getDeclaredConstructors();
314339

315340
/**
316341
* Returns an array reflecting all the methods declared by this type. This method is similar to
317-
* {@link Class#getDeclaredMethods()} in terms of returned methods.
342+
* {@link Class#getDeclaredMethods()} in terms of returned methods. Calling this method forces
343+
* this type to be {@link #link linked}.
318344
*/
319345
ResolvedJavaMethod[] getDeclaredMethods();
320346

‎test/hotspot/jtreg/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestResolvedJavaType.java

+109
Original file line numberDiff line numberDiff line change
@@ -49,6 +49,9 @@
4949
import static org.junit.Assert.assertNull;
5050
import static org.junit.Assert.assertTrue;
5151

52+
import java.io.DataInputStream;
53+
import java.io.IOException;
54+
import java.io.InputStream;
5255
import java.lang.annotation.Annotation;
5356
import java.lang.invoke.MethodHandles.Lookup;
5457
import java.lang.reflect.AccessibleObject;
@@ -64,6 +67,7 @@
6467
import java.util.Map;
6568
import java.util.Set;
6669

70+
import org.junit.Assert;
6771
import org.junit.Test;
6872

6973
import jdk.internal.org.objectweb.asm.*;
@@ -335,6 +339,111 @@ public void findLeastCommonAncestorTest() {
335339
}
336340
}
337341

342+
@Test
343+
public void linkTest() {
344+
for (Class<?> c : classes) {
345+
ResolvedJavaType type = metaAccess.lookupJavaType(c);
346+
type.link();
347+
}
348+
}
349+
350+
private class HidingClassLoader extends ClassLoader {
351+
@Override
352+
protected Class<?> findClass(final String name) throws ClassNotFoundException {
353+
if (name.endsWith("MissingInterface")) {
354+
throw new ClassNotFoundException("missing");
355+
}
356+
byte[] classData = null;
357+
try {
358+
InputStream is = HidingClassLoader.class.getResourceAsStream("/" + name.replace('.', '/') + ".class");
359+
classData = new byte[is.available()];
360+
new DataInputStream(is).readFully(classData);
361+
} catch (IOException e) {
362+
Assert.fail("can't access class: " + name);
363+
}
364+
365+
return defineClass(null, classData, 0, classData.length);
366+
}
367+
368+
ResolvedJavaType lookupJavaType(String name) throws ClassNotFoundException {
369+
return metaAccess.lookupJavaType(loadClass(name));
370+
}
371+
372+
HidingClassLoader() {
373+
super(null);
374+
}
375+
376+
}
377+
378+
interface MissingInterface {
379+
}
380+
381+
static class MissingInterfaceImpl implements MissingInterface {
382+
}
383+
384+
interface SomeInterface {
385+
default MissingInterface someMethod() {
386+
return new MissingInterfaceImpl();
387+
}
388+
}
389+
390+
static class Wrapper implements SomeInterface {
391+
}
392+
393+
@Test
394+
public void linkExceptionTest() throws ClassNotFoundException {
395+
HidingClassLoader cl = new HidingClassLoader();
396+
ResolvedJavaType inner = cl.lookupJavaType(Wrapper.class.getName());
397+
assertTrue("expected default methods", inner.hasDefaultMethods());
398+
try {
399+
inner.link();
400+
assertFalse("link should throw an exception", true);
401+
} catch (NoClassDefFoundError e) {
402+
}
403+
}
404+
405+
@Test
406+
public void hasDefaultMethodsTest() {
407+
for (Class<?> c : classes) {
408+
ResolvedJavaType type = metaAccess.lookupJavaType(c);
409+
assertEquals(hasDefaultMethods(type), type.hasDefaultMethods());
410+
}
411+
}
412+
413+
@Test
414+
public void declaresDefaultMethodsTest() {
415+
for (Class<?> c : classes) {
416+
ResolvedJavaType type = metaAccess.lookupJavaType(c);
417+
assertEquals(declaresDefaultMethods(type), type.declaresDefaultMethods());
418+
}
419+
}
420+
421+
private static boolean hasDefaultMethods(ResolvedJavaType type) {
422+
if (!type.isInterface() && type.getSuperclass() != null && hasDefaultMethods(type.getSuperclass())) {
423+
return true;
424+
}
425+
for (ResolvedJavaType iface : type.getInterfaces()) {
426+
if (hasDefaultMethods(iface)) {
427+
return true;
428+
}
429+
}
430+
return declaresDefaultMethods(type);
431+
}
432+
433+
static boolean declaresDefaultMethods(ResolvedJavaType type) {
434+
if (!type.isInterface()) {
435+
/* Only interfaces can declare default methods. */
436+
return false;
437+
}
438+
for (ResolvedJavaMethod method : type.getDeclaredMethods()) {
439+
if (method.isDefault()) {
440+
assert !Modifier.isStatic(method.getModifiers()) : "Default method that is static?";
441+
return true;
442+
}
443+
}
444+
return false;
445+
}
446+
338447
private static class Base {
339448
}
340449

0 commit comments

Comments
 (0)
Please sign in to comment.