RFR(XL): 6312651: Compiler should only use verified interface types for optimization

Vladimir Ivanov vladimir.x.ivanov at oracle.com
Wed Jul 3 13:19:28 UTC 2019


Hi Roland,

>>>> (2) Numerous test failures:
>>>> #  Internal Error (src/hotspot/share/opto/type.cpp:3696), pid=75399,
>>>> tid=43267
>>>> #  assert(!ik->is_interface()) failed: no interface here
>>>
>>> Does it fail that way with a test I can run?
>>
>> For example, gc/epsilon/TestArraycopyCheckcast.java
> 
> I still can't reproduce that one. Can you give me a short list of tests
> that fail that way?

Just a random selection from failure list:
   test/hotspot/jtreg/runtime/SelectionResolution/InvokeVirtualICCE.java
   compiler/c2/cr6714694/Tester.java
   serviceability/tmtools/jstat/GcTest02.java
   compiler/c2/cr6663854/Test6663854.java

I believe the culprit here is that I observe the failures on MacOS only. 
I have no idea why it doesn't show up on Linux.

>>>> (4) runtime/Unsafe/AllocateInstance.java
>>>> java.lang.AssertionError: Should throw InstantiationException for an
>>>> interface
>>>> 	at AllocateInstance.testInterface(AllocateInstance.java:71)
>>>> 	at AllocateInstance.main(AllocateInstance.java:81)
>>>> 	at
>>>> java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke0(Native
>>>> Method)
>>>> 	at
>>>> java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
>>>> 	at
>>>> java.base/jdk.internal.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
>>>> 	at java.base/java.lang.reflect.Method.invoke(Method.java:567)
>>>> 	at
>>>> com.sun.javatest.regtest.agent.MainActionHelper$AgentVMRunnable.run(MainActionHelper.java:298)
>>>> 	at java.base/java.lang.Thread.run(Thread.java:830)
>>>
>>> That one runs fine with no option or -Xcomp. Does it need any specific
>>> option to fail?
>>
>> It fails reliably w/o any additional flags.
> 
> I can reproduce it now. It seems there's a bigger problem
> though. Current C2 compiles the following method:
> 
>      private static void test3(Class klass) throws IllegalAccessException, InstantiationException {
>          UNSAFE.allocateInstance(klass);
>      }
> 
> without any allocation (EA removes it). Calling it with an interface:
> 
> test3(I.class);
> 
> then succeeds. So either we trust the unsafe user to know what she/he's
> doing or we need a runtime check (that would also cover the failure you
> reported with my patch). What do you think?

There's a similar bug filed against Unsafe for array classes 
(JDK-8153540 [1]), but I think we see a different problem here, since 
there are already proper checks in-place for interfaces (slow path on 
Klass::_layout_helper check for _lh_instance_slow_path_bit).

As I mentioned, I see the root cause as Object vs interface class 
reported as exact class. It breaks the logic in 2 places: EA and 
allocation logic.


(1) src/hotspot/share/opto/escape.cpp
-      if (cik->is_subclass_of(_compile->env()->Thread_klass()) ||
-          cik->is_subclass_of(_compile->env()->Reference_klass()) ||
-         !cik->is_instance_klass() || // StressReflectiveCode
-         !cik->as_instance_klass()->can_be_instantiated() ||
-          cik->as_instance_klass()->has_finalizer()) {
+      ciInstanceKlass* ik = kt->is_instklassptr()->instance_klass();
+      if (ik->is_subclass_of(_compile->env()->Thread_klass()) ||
+          ik->is_subclass_of(_compile->env()->Reference_klass()) ||
+          !ik->can_be_instantiated() ||
+          ik->has_finalizer()) {

cik was interface class, ik is Object now and ik->can_be_instantiated() 
becomes true, which is wrong.


(2) src/hotspot/share/opto/graphKit.cpp

@@ -3365,14 +3385,22 @@
  // This two-faced routine is useful because allocation sites
  // almost always feature constant types.
  Node* GraphKit::get_layout_helper(Node* klass_node, jint& 
constant_value) {
    const TypeKlassPtr* inst_klass = _gvn.type(klass_node)->isa_klassptr();
    if (!StressReflectiveCode && inst_klass != NULL) {
-    ciKlass* klass = inst_klass->klass();
      bool    xklass = inst_klass->klass_is_exact();
-    if (xklass || klass->is_array_klass()) {
-      jint lhelper = klass->layout_helper();
+    if (xklass || inst_klass->isa_aryklassptr()) {
+      jint lhelper;
+      if (inst_klass->isa_aryklassptr()) {
+        BasicType elem = 
inst_klass->as_instance_type()->isa_aryptr()->elem()->array_element_basic_type();
+        if (elem == T_ARRAY || elem == T_NARROWOOP) {
+          elem = T_OBJECT;
+        }
+        lhelper = Klass::array_layout_helper(elem);
+      } else {
+        lhelper = 
inst_klass->is_instklassptr()->instance_klass()->layout_helper();
+      }
        if (lhelper != Klass::_lh_neutral_value) {
          constant_value = lhelper;
          return (Node*) NULL;
        }
      }

GraphKit::get_layout_helper() returns layout for java.lang.Object 
instead of AnInterface ([2] vs [3]) and that folds away the slow path 
(layout_con=17/must_go_slow=1 vs layout_con=16/must_go_slow=0).

That's why even if allocation is preserved, allocation fast path (in 
TLAB) misses proper check with your patch.

Best regards,
Vladimir Ivanov

[1] https://bugs.openjdk.java.net/browse/JDK-8153540

[2] Patched: from GraphKit::get_layout_helper:
(lldb) p inst_klass->dump(); tty->cr()
klass java/lang/Object: 0x0000000105828a20 
(AllocateInstance$AnInterface)precise :Constant:exact *

(lldb) p inst_klass->is_instklassptr()->instance_klass()->print() 
<ciInstanceKlass name=java/lang/Object loader=0x0000000000000000 
loaded=true initialized=true finalized=false subklass=true size=16 
flags=public,super ident=1016 address=0x0000000105828a20>

(lldb) p inst_klass->is_instklassptr()->instance_klass()->layout_helper()
16

(lldb) p inst_klass->exact_klass(false)->print()
<ciInstanceKlass name=AllocateInstance$AnInterface 
loader=0x000000070ff165e8 loaded=true initialized=false finalized=false 
subklass=false size=17 flags=DEFAULT_ACCESS,interface,abstract 
mirror=PRESENT ident=1158 address=0x0000000100891f50>

(lldb) p inst_klass->exact_klass(false)->layout_helper()
17


   static bool layout_helper_needs_slow_path(jint lh) {
     assert(lh > (jint)_lh_neutral_value, "must be instance");
     return (lh & _lh_instance_slow_path_bit) != 0;
   }

     _lh_instance_slow_path_bit  = 0x01,



[3] Original behavior: from GraphKit::get_layout_helper:
(lldb) p klass_node->dump()
  27	ConP	===  0  [[ 29  29 ]]  #precise klass 
AllocateInstance$AnInterface: 0x0000000100892d50:Constant:exact * 
Interface:precise klass AllocateInstance$AnInterface: 
0x0000000100892d50:Constant:exact *

(lldb) p inst_klass->dump()
precise klass AllocateInstance$AnInterface: 
0x0000000100892d50:Constant:exact *

(lldb) p xklass
true

(lldb) p lhelper
17


More information about the hotspot-compiler-dev mailing list