<div dir="ltr">Hi Ioi,<div><br></div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex">I was having a similar circularity issue (but in production time) and I just added enough classes to make the NPE go away.<br></blockquote><div> </div><div><div>I am wondering if you have a test case that reproduces the NPE which prompted you to add these classes:</div><div><br></div><div><a href="https://github.com/openjdk/leyden/blob/7781109154bf2af89854c7e13aa3e160bb82608e/src/hotspot/share/cds/aotClassInitializer.cpp#L65-L78">https://github.com/openjdk/leyden/blob/7781109154bf2af89854c7e13aa3e160bb82608e/src/hotspot/share/cds/aotClassInitializer.cpp#L65-L78</a></div><div><br></div><div>I commented this code and ran the tests under premain but didn't hit the NPE.</div><div><br></div><div>Thanks,</div><div><div><div dir="ltr" class="gmail_signature" data-smartmail="gmail_signature"><div dir="ltr">- Ashutosh Mehra</div></div></div><br></div></div></div><br><div class="gmail_quote"><div dir="ltr" class="gmail_attr">On Tue, Sep 17, 2024 at 1:11 AM <<a href="mailto:ioi.lam@oracle.com">ioi.lam@oracle.com</a>> wrote:<br></div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex"><u></u>

  
  <div>
    <p>Hi Ashutosh,<br>
    </p>
    <p>So this looks like a potential bug (or feature) in the core lib
      code. When CDS forcefully initializes a class in an unexpected (or
      untested) order, the "initialization soup" fails.</p>
    <p>Perhaps a work-around would be to make some harmless calls at the
      place where CDS was calling into
      MethodType.createArchivedObjects(). E.g., do something like this:</p>
    <p><font face="monospace">+    if
        (CDSConfig::is_dumping_invokedynamic()) {<br>
        +       // call into Java:
        jdk.internal.misc::warmupInvokeDynamic();<br>
        +   }<br>
            // Rewrite and link classes<br>
            log_info(cds)("Rewriting and linking classes ...");<br>
        <br>
      </font>Maybe you can add a new Lambda expressions, MethodHandle
      invocations, etc, that would hopefully cause
      PrimitiveClassDescImpl and friends to be initialized in their
      natural order.<br>
    </p>
    <p>Or call class.forName("java.lang.constant.ConstantDescs") ??</p>
    <p>BTW, you can see my comments in
      AOTClassInitializer::is_forced_preinit_class():<br>
      <br>
          // TODO:<br>
          // This is needed since JDK-8338532. Without this, when<br>
          // archived heap objects are used, the class init order is not<br>
          // expected by the jdk/internal/constant bootstrap code and we<br>
          // will get a null pointer exception.<br>
          //<br>
          // When bootstraping has intricated/fragile order, it's
      probably<br>
          // better to archive all related classes in an initialized
      state<br>
          // (i.e., take a snapshot). The existing approach in<br>
          // heapShared::resolve_or_init_classes_for_subgraph_of() won't
      work.<br>
          "jdk/internal/constant/PrimitiveClassDescImpl",<br>
          "jdk/internal/constant/ReferenceClassDescImpl",<br>
          "java/lang/constant/ConstantDescs",<br>
          "sun/invoke/util/Wrapper",<br>
      <br>
      I was having a similar circularity issue (but in production time)
      and I just added enough classes to make the NPE go away. For your
      test case, if you manage to fix in in the assembly run but run
      into NPE in production run, you might need to add more classes to
      this list. Yes, it's a hack :-(<br>
    </p>
    <p><br>
    </p>
    <p>Thanks</p>
    <p>- Ioi<br>
    </p>
    <div>On 9/13/24 7:05 PM, Ashutosh Mehra
      wrote:<br>
    </div>
    <blockquote type="cite">
      
      <div dir="ltr">This is turning out to be a real example of class
        initialization soup!
        <div>As mentioned during the meeting, I am getting NPE in the
          assembly phase when testing the patch [0] that I proposed in
          my earlier mail </div>
        <div>using a test case inspired by the Infinispan code.</div>
        <div>NPE occurs when running the class initializer for <font face="monospace">PrimitiveClassDescImpl </font></div>
        <div>
          <div>Interestingly, <span style="font-family:monospace">PrimitiveClassDescImpl</span> is
            "forcefully" initialized by <font face="monospace">MetaspaceShared::link_shared_classes()</font>.</div>
          <div><br>
          </div>
          <div>
            <div>
              <div dir="ltr" class="gmail_signature">
                <div dir="ltr">I couldn't get a stack trace so I relied
                  on exception logs (using -Xlog:exceptions=trace) to
                  find the cause which indicate following frames on the
                  stack:</div>
                <div dir="ltr"><br>
                </div>
                <div dir="ltr"><font face="monospace">[0]
jdk/internal/constant/MethodTypeDescImpl::validateArgument(Ljava/lang/constant/ClassDesc;)Ljava/lang/constant/ClassDesc; @
                    bci 1<br>
                  </font></div>
                <div dir="ltr"><font face="monospace">[1]
jdk/internal/constant/MethodTypeDescImpl::ofTrusted(Ljava/lang/constant/ClassDesc;[Ljava/lang/constant/ClassDesc;)Ljdk/internal/constant/MethodTypeDescImpl; @
                    bci 27<br>
                  </font></div>
                <div dir="ltr"><font face="monospace">[2]
java/lang/constant/ConstantDescs::ofConstantBootstrap(Ljava/lang/constant/ClassDesc;Ljava/lang/String;Ljava/lang/constant/ClassDesc;[Ljava/lang/constant/ClassDesc;)Ljava/lang/constant/DirectMethodHandleDesc; @
                    bci 47</font></div>
                <div dir="ltr"><font face="monospace">[3]
                    java/lang/constant/ConstantDescs::<clinit> @
                    bci 664<br>
                  </font></div>
                <div dir="ltr"><font face="monospace">[4]
jdk/internal/constant/PrimitiveClassDescImpl::<init>(Ljava/lang/String;)V @
                    bci 1<br>
                  </font></div>
                <div dir="ltr"><font face="monospace">[5]
jdk/internal/constant/PrimitiveClassDescImpl::<clinit>(Ljava/lang/String;)V @
                    bci 6</font><span style="font-family:monospace"><br>
                  </span></div>
                <div dir="ltr"><span style="font-family:monospace"><br>
                  </span></div>
                <div dir="ltr">Notice that invocation of <font face="monospace">PrimitiveClassDescImpl::<clinit>
                  </font>results in initialization of <font face="monospace">ConstantDescs</font> class (see
                  frame 3).<br>
                </div>
                <div><font face="monospace">ConstantDescs::<clinit> @
                    664 </font><font face="arial, sans-serif">corresponds
                    to following java code:</font></div>
                <div><br>
                </div>
                <div><font face="monospace">    public static final
                    DirectMethodHandleDesc BSM_CLASS_DATA_AT<br>
                  </font></div>
                <div><font face="monospace">            =
                    ofConstantBootstrap(CD_MethodHandles, "classDataAt",<br>
                                CD_Object, CD_int);<br>
                  </font></div>
                <div><br>
                </div>
                <div>The last parameter CD_int is initialized as:</div>
                <div><br>
                </div>
                <div><font face="monospace">    public static final
                    ClassDesc CD_int = PrimitiveClassDescImpl.CD_int;<br>
                  </font></div>
                <div><br>
                </div>
                <div>So, its value is obtained from <font face="monospace">PrimitiveClassDescImpl.CD_int</font> which
                  hasn't been initialized properly yet. As a result <font face="monospace">ConstantDescs::CD_int </font><font face="arial, sans-serif">is assigned</font> null
                  which results in <font face="monospace">MethodTypeDescImpl::validateArgument</font>
                  throwing NPE later.</div>
                <div>There is a clear class initialization circularity
                  involving <font face="monospace">PrimitiveClassDescImpl</font>
                  and <font face="monospace">ConstantDescs</font>, and
                  the result depends on which class gets initialized
                  first.</div>
                <div><br>
                </div>
                <div>Without my patch this issue is not seen because <font face="monospace">PrimitiveClassDescImpl</font> has
                  already been initialized by the time <font face="monospace">MetaspaceShared::link_shared_classes()</font><font face="arial, sans-serif"> is called.</font></div>
                <div><font face="arial, sans-serif">Its initialization
                    is triggered by the call to </font><font face="monospace">MethodType::createArchivedObjects(). </font></div>
                <div><font face="arial, sans-serif">It also explains why
                    my patch introduced this issue because it
                    effectively moved the call to </font><span style="font-family:monospace">MethodType::createArchivedObjects()
                  </span><font face="arial, sans-serif">after </font><span style="font-family:monospace">MetaspaceShared::link_shared_classes().</span></div>
                <div><span style="font-family:monospace"><br>
                  </span></div>
                <div><font face="arial, sans-serif">[0] <a href="https://urldefense.com/v3/__https://github.com/ashu-mehra/leyden/commit/d8f99cce67df1c7b0f7ef8562676df438633a66e__;!!ACWV5N9M2RV99hQ!MnS3LSY2PXCCXbHvdyMfR6Nj57XH7Ey7aZuOLmQ__hhE_RhUrKnkzZ0uP8AwLZYcl8lAhhZK3GwoiA$" target="_blank">https://github.com/ashu-mehra/leyden/commit/d8f99cce67df1c7b0f7ef8562676df438633a66e</a></font></div>
                <div><font face="arial, sans-serif"><br>
                  </font></div>
                <div><font face="arial, sans-serif">Thanks,</font></div>
                <div><font face="arial, sans-serif">- Ashutosh Mehra</font></div>
              </div>
            </div>
            <br>
          </div>
        </div>
      </div>
      <br>
      <div class="gmail_quote">
        <div dir="ltr" class="gmail_attr">On Wed, Sep 11, 2024 at
          3:12 PM Ashutosh Mehra <<a href="mailto:asmehra@redhat.com" target="_blank">asmehra@redhat.com</a>>
          wrote:<br>
        </div>
        <blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex">
          <div dir="ltr">Regarding the WrongMethodTypeException that I
            mentioned in my previous email (see pt 3),
            <div>this exception happens when lambda proxy class attempts
              to invoke the MethodHandle it obtained from the classData:</div>
            <div><br>
            </div>
            <div><font face="monospace">  public void
                accept(java.lang.Object);<br>
                    descriptor: (Ljava/lang/Object;)V<br>
                    flags: (0x0001) ACC_PUBLIC<br>
                    Code:<br>
                      stack=3, locals=2, args_size=2<br>
                         0: ldc           #26                 // Dynamic
                #0:_:Ljava/lang/invoke/MethodHandle;<br>
                         2: aload_0<br>
                         3: getfield      #13                 // Field
                arg$1:Lorg/wildfly/security/WildFlyElytronBaseProvider;<br>
                         6: aload_1<br>
                         7: checkcast     #28                 // class
                java/security/Provider$Service<br>
                        10: invokevirtual #34                 // Method
java/lang/invoke/MethodHandle.invokeExact:(Lorg/wildfly/security/WildFlyElytronBaseProvider;Ljava/security/Provider$Service;)V<br>
                        13: return</font><br>
            </div>
            <div><br>
            </div>
            <div>
              <div>The scenario is during the assembly phase as part of
                the indy resolution the MethodHandle for which the
                exception is thrown gets created.</div>
              <div>Normally MethodHandle's type gets added in <font face="monospace">MethodType::internTable</font> but by
                the time indy resolution happens, JVM has already taken </div>
              <div>snapshot of the <font face="monospace">MethodType::internTable</font>
                through an upcall to <font face="monospace">MethodType::createArchivedObjects()</font>.</div>
              <div>As a result the AOTCache ends up with the MethodType
                object which is not in <font face="monospace">AOTHolder.archivedMethodTypes</font>.</div>
              <div><br>
              </div>
              <div>During the production run, when the jvm invokes the
                MethodHandle, it searches for the MethodType
                corresponding to the signature passed at the callsite.</div>
              <div>As expected, it fails to find it in the <font face="monospace">AOTHolder.archivedMethodTypes</font>,
                so it creates a new instance of the MethodType.</div>
              <div>But <font face="monospace">Invokers.checkExactType()</font>
                relies on the MethodHandle's type to be the same object
                as the MethodType object passed as parameter.</div>
              <div><br>
              </div>
              <div><font face="monospace">    static void
                  checkExactType(M</font><span style="font-family:monospace">ethodHandle mh</span><span style="font-family:monospace">M, MethodType expected)
                  {</span></div>
              <div><font face="monospace">        MethodType targetType
                  = mh.type();<br>
                          if (targetType != expected)<br>
                              throw
                  newWrongMethodTypeException(targetType, expected);<br>
                      }</font><br>
              </div>
              <div><br>
              </div>
              <div>Hence, it throws <span style="font-family:monospace">WrongMethodTypeException</span><font face="arial, sans-serif"> though the two MT objects
                  have the same signature.</font></div>
              <div><font face="arial, sans-serif"><br>
                </font></div>
              <div><font face="arial, sans-serif">To handle this
                  scenario, I changed the order of indy resolution and </font>upcall
                to <font face="monospace">MethodType::createArchivedObjects()</font><font face="arial, sans-serif"> as:</font></div>
              <div><font face="arial, sans-serif"><br>
                </font></div>
              <div><font face="monospace">diff --git
                  a/src/hotspot/share/cds/metaspaceShared.cpp
                  b/src/hotspot/share/cds/metaspaceShared.cpp<br>
                  index df4bcadefa3..457716cac5b 100644<br>
                  --- a/src/hotspot/share/cds/metaspaceShared.cpp<br>
                  +++ b/src/hotspot/share/cds/metaspaceShared.cpp<br>
                  @@ -751,6 +751,20 @@ void
                  MetaspaceShared::link_shared_classes(bool
                  jcmd_request, TRAPS) {<br>
                     if (CDSConfig::is_dumping_final_static_archive()) {<br>
                       FinalImageRecipes::apply_recipes(CHECK);<br>
                     }<br>
                  +<br>
                  +#if INCLUDE_CDS_JAVA_HEAP<br>
                  +  if (CDSConfig::is_dumping_invokedynamic()) {<br>
                  +    // This makes sure that the MethodType and
                  MethodTypeForm tables won't be updated<br>
                  +    // concurrently when we are saving their contents
                  into a side table.<br>
                  +  
                   assert(CDSConfig::allow_only_single_java_thread(),
                  "Required");<br>
                  +<br>
                  +    JavaValue result(T_VOID);<br>
                  +    JavaCalls::call_static(&result,
                  vmClasses::MethodType_klass(),<br>
                  +                          
                  vmSymbols::createArchivedObjects(),<br>
                  +                          
                  vmSymbols::void_method_signature(),<br>
                  +                           CHECK);<br>
                  +  }<br>
                  +#endif<br>
                   }</font><br>
                 </div>
              <div>Note that indy resolution happens as part of <span style="font-family:monospace">FinalImageRecipes::apply_recipes(CHECK)
                </span><font face="arial, sans-serif">which is now
                  invoked before the upcall to </font><span style="font-family:monospace">createArchivedObjects().</span></div>
              <div><font face="arial, sans-serif">With this change I am
                  able to run the application without any exceptions.</font></div>
              <div>My complete patch can be seen here: <a href="https://urldefense.com/v3/__https://github.com/ashu-mehra/leyden/commit/d8f99cce67df1c7b0f7ef8562676df438633a66e__;!!ACWV5N9M2RV99hQ!MnS3LSY2PXCCXbHvdyMfR6Nj57XH7Ey7aZuOLmQ__hhE_RhUrKnkzZ0uP8AwLZYcl8lAhhZK3GwoiA$" target="_blank">https://github.com/ashu-mehra/leyden/commit/d8f99cce67df1c7b0f7ef8562676df438633a66e</a></div>
              <div>I will do more testing with this patch.</div>
              <div><br>
              </div>
              <div><a class="gmail_plusreply" id="m_-6081196208708063630m_-8326742188757259955m_-669397441264475055m_4538222356731470316m_-1862706623606286828plusReplyChip-0" href="mailto:ioi.lam@oracle.com" target="_blank">@Ioi Lam</a> do you have any
                feedback on this patch.<br>
              </div>
              <div><br>
              </div>
              <div>Thanks,<br clear="all">
                <div>
                  <div dir="ltr" class="gmail_signature">
                    <div dir="ltr">- Ashutosh Mehra</div>
                  </div>
                </div>
                <br>
              </div>
              <div><br>
              </div>
            </div>
          </div>
          <br>
          <div class="gmail_quote">
            <div dir="ltr" class="gmail_attr">On Wed, Sep 11, 2024 at
              10:14 AM Ashutosh Mehra <<a href="mailto:asmehra@redhat.com" target="_blank">asmehra@redhat.com</a>>
              wrote:<br>
            </div>
            <blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex">
              <div dir="ltr">Hi Andrew,
                <div>Thanks for sharing the initial investigation.
                  <div>I have been looking into this and have a few of
                    things to add to your analysis:</div>
                  <div><br>
                  </div>
                  <div>1.  As you mentioned the classData for the lambda
                    class WildFlyElytronBaseProvider$$Lambda is null.</div>
                  <div>The classData is stored in the mirror object of
                    the InstanceKlass when the class is defined
                    through JVM_LookupDefineClass.</div>
                  <div>However, when we create the scratch mirror object
                    (which get stored in the AOT cache) the classData is
                    not populated.</div>
                  <div>See <a href="https://urldefense.com/v3/__https://github.com/openjdk/leyden/blob/d23b9f2d5e3523cc547337da59327ed86a6057a3/src/hotspot/share/classfile/javaClasses.cpp*L1128-L1131__;Iw!!ACWV5N9M2RV99hQ!MnS3LSY2PXCCXbHvdyMfR6Nj57XH7Ey7aZuOLmQ__hhE_RhUrKnkzZ0uP8AwLZYcl8lAhhbKjzG3gw$" target="_blank">https://github.com/openjdk/leyden/blob/d23b9f2d5e3523cc547337da59327ed86a6057a3/src/hotspot/share/classfile/javaClasses.cpp#L1128-L1131</a></div>
                  <div><br>
                  </div>
                  <div><font face="monospace">  Handle classData; // set
                      to null. Will be reinitialized at runtime<br>
                        Handle mirror;<br>
                        Handle comp_mirror;<br>
                        allocate_mirror(k, /*is_scratch=*/true,
                      protection_domain, classData, mirror, comp_mirror,
                      CHECK);</font><br>
                  </div>
                  <div><br>
                  </div>
                  <div>So this explains why the call to <font face="monospace">classData(caller.lookupClass())</font><font face="arial, sans-serif"> returned null.</font></div>
                  <div><br>
                  </div>
                  <div>2. In the mainline there is a check
                    in InnerClassLambdaMetafactory.java for the
                    particular code pattern used by the application.</div>
                  <div>If this code pattern is found then the lambda
                    proxy class is not included in the CDS archive.</div>
                  <div>See <a href="https://urldefense.com/v3/__https://github.com/openjdk/leyden/blob/d23b9f2d5e3523cc547337da59327ed86a6057a3/src/java.base/share/classes/java/lang/invoke/InnerClassLambdaMetafactory.java*L163-L170__;Iw!!ACWV5N9M2RV99hQ!MnS3LSY2PXCCXbHvdyMfR6Nj57XH7Ey7aZuOLmQ__hhE_RhUrKnkzZ0uP8AwLZYcl8lAhhZl2S9tQg$" target="_blank">https://github.com/openjdk/leyden/blob/d23b9f2d5e3523cc547337da59327ed86a6057a3/src/java.base/share/classes/java/lang/invoke/InnerClassLambdaMetafactory.java#L163-L170</a></div>
                  <div>and <a href="https://urldefense.com/v3/__https://github.com/openjdk/leyden/blob/d23b9f2d5e3523cc547337da59327ed86a6057a3/src/java.base/share/classes/java/lang/invoke/InnerClassLambdaMetafactory.java*L246__;Iw!!ACWV5N9M2RV99hQ!MnS3LSY2PXCCXbHvdyMfR6Nj57XH7Ey7aZuOLmQ__hhE_RhUrKnkzZ0uP8AwLZYcl8lAhhYsI3Xx0g$" target="_blank">https://github.com/openjdk/leyden/blob/d23b9f2d5e3523cc547337da59327ed86a6057a3/src/java.base/share/classes/java/lang/invoke/InnerClassLambdaMetafactory.java#L246</a></div>
                  <div><br>
                  </div>
                  <div><font face="monospace">        // If the target
                      class invokes a protected method inherited from a<br>
                              // superclass in a different package, or
                      does 'invokespecial', the<br>
                              // lambda class has no access to the
                      resolved method, or does<br>
                              // 'invokestatic' on a hidden class which
                      cannot be resolved by name.<br>
                              // Instead, we need to pass the live
                      implementation method handle to<br>
                              // the proxy class to invoke directly.
                      (javac prefers to avoid this<br>
                              // situation by generating bridges in the
                      target class)<br>
                              useImplMethodHandle =
                      (Modifier.isProtected(implInfo.getModifiers())
                      &&<br>
                                                   
                       !VerifyAccess.isSamePackage(targetClass,
                      implInfo.getDeclaringClass())) ||<br>
                                                     implKind ==
                      MethodHandleInfo.REF_invokeSpecial ||<br>
                                                     implKind ==
                      MethodHandleInfo.REF_invokeStatic &&
                      implClass.isHidden();</font><br>
                  </div>
                  <div><br>
                  </div>
                  <div>In premain lambda proxy classes get included in
                    the AOT cache as a result of indy resolution and
                    that mechanism doesn't have this kind of check.</div>
                  <div><br>
                  </div>
                  <div>3. For the null classData problem mentioned
                    above, I tried to fix it by storing classData in the
                    scratch mirror using the following patch:<br>
                    <br>
                  </div>
                  <div><font face="monospace">diff --git
                      a/src/hotspot/share/classfile/javaClasses.cpp
                      b/src/hotspot/share/classfile/javaClasses.cpp<br>
                      index bd8141adbcc..41766e98093 100644<br>
                      --- a/src/hotspot/share/classfile/javaClasses.cpp<br>
                      +++ b/src/hotspot/share/classfile/javaClasses.cpp<br>
                      @@ -1094,9 +1094,9 @@ void
                      java_lang_Class::create_mirror(Klass* k, Handle
                      class_loader,<br>
                           }<br>
                           if (CDSConfig::is_dumping_heap()) {<br>
                             if
                      (CDSConfig::is_dumping_protection_domains()) {<br>
                      -        create_scratch_mirror(k,
                      protection_domain, CHECK);<br>
                      +        create_scratch_mirror(k,
                      protection_domain, classData, CHECK);<br>
                             } else {<br>
                      -        create_scratch_mirror(k, Handle() /* null
                      protection_domain*/, CHECK);<br>
                      +        create_scratch_mirror(k, Handle() /* null
                      protection_domain*/, classData, CHECK);<br>
                             }<br>
                           }<br>
                         } else {<br>
                      @@ -1117,7 +1117,7 @@ void
                      java_lang_Class::create_mirror(Klass* k, Handle
                      class_loader,<br>
                       // Note: we archive the "scratch mirror" instead
                      of k->java_mirror(), because the<br>
                       // latter may contain dumptime-specific
                      information that cannot be archived<br>
                       // (e.g., ClassLoaderData*, or static fields that
                      are modified by Java code execution).<br>
                      -void
                      java_lang_Class::create_scratch_mirror(Klass* k,
                      Handle protection_domain, TRAPS) {<br>
                      +void
                      java_lang_Class::create_scratch_mirror(Klass* k,
                      Handle protection_domain, Handle classData, TRAPS)
                      {<br>
                         if (k->class_loader() != nullptr &&<br>
                             k->class_loader() !=
                      SystemDictionary::java_platform_loader()
                      &&<br>
                             k->class_loader() !=
                      SystemDictionary::java_system_loader()) {<br>
                      @@ -1125,9 +1125,11 @@ void
                      java_lang_Class::create_scratch_mirror(Klass* k,
                      Handle protection_domain,<br>
                           return;<br>
                         }<br>
                       <br>
                      -  Handle classData; // set to null. Will be
                      reinitialized at runtime<br>
                      +  //Handle classData; // set to null. Will be
                      reinitialized at runtime<br>
                         Handle mirror;<br>
                         Handle comp_mirror;<br>
                         allocate_mirror(k, /*is_scratch=*/true,
                      protection_domain, classData, mirror, comp_mirror,
                      CHECK);<br>
                       <br>
                         if (comp_mirror() != nullptr) {<br>
                      diff --git
                      a/src/hotspot/share/classfile/javaClasses.hpp
                      b/src/hotspot/share/classfile/javaClasses.hpp<br>
                      index bc49a0861a7..7ec2a2556dd 100644<br>
                      --- a/src/hotspot/share/classfile/javaClasses.hpp<br>
                      +++ b/src/hotspot/share/classfile/javaClasses.hpp<br>
                      @@ -263,7 +263,7 @@ class java_lang_Class :
                      AllStatic {<br>
                       <br>
                         // Archiving<br>
                         static void serialize_offsets(SerializeClosure*
                      f) NOT_CDS_RETURN;<br>
                      -  static void create_scratch_mirror(Klass* k,
                      Handle protection_domain, TRAPS)
                      NOT_CDS_JAVA_HEAP_RETURN;<br>
                      +  static void create_scratch_mirror(Klass* k,
                      Handle protection_domain, Handle classData, TRAPS)
                      NOT_CDS_JAVA_HEAP_RETURN;<br>
                         static bool restore_archived_mirror(Klass *k,
                      Handle class_loader, Handle module,<br>
                                                             Handle
                      protection_domain,<br>
                                                             TRAPS)
                      NOT_CDS_JAVA_HEAP_RETURN_(false);</font><br>
                  </div>
                  <div><br>
                  </div>
                  <div>But this resulted in a different exception:</div>
                  <div><br>
                  </div>
                  <div><font face="monospace">Exception in thread "main"
                      java.lang.ExceptionInInitializerError<br>
                      at com.redhat.leyden.Main.main(Main.java:7)<br>
                      Caused by:
                      java.lang.invoke.WrongMethodTypeException:
                      handle's method type
                      (WildFlyElytronBaseProvider,Service)void but found
                      (WildFlyElytronBaseProvider,Service)void<br>
                      at
java.base/java.lang.invoke.Invokers.newWrongMethodTypeException(Invokers.java:521)<br>
                      at
                      java.base/java.lang.invoke.Invokers.checkExactType(Invokers.java:530)<br>
                      at
java.base/java.lang.invoke.Invokers$Holder.invokeExact_MT(Invokers$Holder)<br>
                      at
org.wildfly.security.WildFlyElytronBaseProvider$$Lambda/0x80000000c.accept(Unknown
                      Source)<br>
                      at
org.wildfly.security.WildFlyElytronBaseProvider.putMakedPasswordImplementations(WildFlyElytronBaseProvider.java:112)<br>
                      at
org.wildfly.security.WildFlyElytronBaseProvider.putPasswordImplementations(WildFlyElytronBaseProvider.java:107)<br>
                      at
org.wildfly.security.password.WildFlyElytronPasswordProvider.<init>(WildFlyElytronPasswordProvider.java:43)<br>
                      at
org.wildfly.security.password.WildFlyElytronPasswordProvider.<clinit>(WildFlyElytronPasswordProvider.java:36)<br>
                      ... 1 more</font><br>
                  </div>
                  <div><font face="monospace"><br>
                    </font></div>
                  <div>The exception message is strange because the
                    handle's method type and the expected type are both
                    symbolically the same.</div>
                  <div>I am debugging this exception at the moment.</div>
                  <div> </div>
                  <div>Thanks,</div>
                  <div>
                    <div>
                      <div dir="ltr" class="gmail_signature">
                        <div dir="ltr">- Ashutosh Mehra</div>
                      </div>
                    </div>
                    <br>
                  </div>
                </div>
              </div>
              <br>
              <div class="gmail_quote">
                <div dir="ltr" class="gmail_attr">On Wed, Sep 11, 2024
                  at 6:03 AM Andrew Dinn <<a href="mailto:adinn@redhat.com" target="_blank">adinn@redhat.com</a>>
                  wrote:<br>
                </div>
                <blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex">Oops,
                  sorry, I debugged this a few days ago! Correction to a
                  few details:<br>
                  <br>
                  On 11/09/2024 10:39, Andrew Dinn wrote:<br>
                  > A crash due to an NPE was observed in the
                  Infinispan (Data Grid) server <br>
                  > app when deployed using the Leyden EA. The crash
                  still manifests with <br>
                  > the latest premain code. The crash happens below
                  an application call <br>
                  > which employs a method reference as argument<br>
                  > <br>
                  >         
                  putMakedPasswordImplementations(this::putService,
                  this);<br>
                  <br>
                  The called method in turn calls consumer.accept<br>
                  <br>
                               consumer.accept(new Service(provider, <br>
                  PASSWORD_FACTORY_TYPE, algorithm, <br>
"org.wildfly.security.password.impl.PasswordFactorySpiImpl", emptyList,
                  <br>
                  emptyMap));<br>
                  <br>
                  which enters enters
                  MethodHandleNative::linkDynamicConstant()<br>
                  <br>
                  > Debugging shows that the call to
                  linkDynamicConstant() returns null.<br>
                  > <br>
                  > A simple reproducer for the problem is available
                  as a maven project on <br>
                  > github:<br>
                  > <br>
                  >    <a href="https://urldefense.com/v3/__https://github.com/tristantarrant/elytron-leyden__;!!ACWV5N9M2RV99hQ!MnS3LSY2PXCCXbHvdyMfR6Nj57XH7Ey7aZuOLmQ__hhE_RhUrKnkzZ0uP8AwLZYcl8lAhhZaZr1akQ$" rel="noreferrer" target="_blank">https://github.com/tristantarrant/elytron-leyden</a><br>
                  > <br>
                  > The ReadMe provides an explanation of how to
                  reproduce the problem. I <br>
                  > did so and the debugged to find out some of the
                  details of what is <br>
                  > happening (see below) but did not fully clarify
                  the problem. Help from <br>
                  > someone more conversant with the ins and outs of
                  method handle <br>
                  > bootstraps in premain would be welcome. Details
                  follow.<br>
                  > <br>
                  > regards,<br>
                  > <br>
                  > <br>
                  > Andrew Dinn<br>
                  > -----------<br>
                  > <br>
                  > I downloaded the git repo and attached the Java
                  sources using Maven command<br>
                  > <br>
                  >    $ mvn dependency:sources<br>
                  > <br>
                  > Having manifested the crash by following the
                  instructions in the README <br>
                  > I reran the leyden JVM under gdb using the
                  following commands to enable <br>
                  > Java debugging<br>
                  > <br>
                  > $ gdb ${LEYDEN_HOME}/bin/java<br>
                  > (gdb) cd /path/to/mvn/project<br>
                  > (gdb) run <br>
                  >
                  -agentlib:jdwp=transport=dt_socket,server=y,suspend=y,address=5005
                  <br>
                  > -classpath <br>
                  >
/home/adinn/redhat/openjdk/infinispan/elytron-leyden/base/target/elytron-leyden-base-0.0.1-SNAPSHOT.jar:/home/adinn/.m2/repository/org/wildfly/security/wildfly-elytron-credential/<a href="https://urldefense.com/v3/__http://2.5.1.__;!!ACWV5N9M2RV99hQ!MnS3LSY2PXCCXbHvdyMfR6Nj57XH7Ey7aZuOLmQ__hhE_RhUrKnkzZ0uP8AwLZYcl8lAhhauJ-RjcQ$" target="_blank">2.5.1.</a>Final/wildfly-elytron-credential-2.5.1.Final.jar:/home/adinn/.m2/repository/org/wildfly/security/wildfly-elytron-base/2.5.1.Final/wildfly-elytron-base-2.5.1.Final.jar
                  -XX:CacheDataStore=elytron.aot com.redhat.leyden.Main<br>
                  > <br>
                  > The problem manifests at
                  WildflyElytronBaseProvider.java:112 in method <br>
                  >
                  WildflyElytronBaseProvider::putMakedPasswordImplementations<br>
                  <br>
                       static void
                  putMakedPasswordImplementations(Consumer<Service>
                  <br>
                  consumer, Provider provider) {<br>
                           for (String algorithm : MASKED_ALGORITHMS) {<br>
                               consumer.accept(new Service(provider, <br>
                  PASSWORD_FACTORY_TYPE, algorithm, <br>
"org.wildfly.security.password.impl.PasswordFactorySpiImpl", emptyList,
                  <br>
                  emptyMap)); <== NPE under this call<br>
                           }<br>
                  <br>
                  <br>
                  > The source code for this method can be found in
                  the following source jar<br>
                  > <br>
                  > <br>
                  >
${M2_REPO}/org/wildfly/security/wildfly-elytron-base/2.5.1.Final/wildfly-elytron-base-2.5.1.Final-sources.jar<br>
                  > <br>
                  > (where M2_REPO will normally be ~/.m2/repository)<br>
                  > <br>
                  > Stepping into accept eventually enters
                  MethodHandleNative::linkDynamicConstant <br>
                  > which in turn enters into
                  ConstantBootstraps.makeConstant(). The caller <br>
                  > Class at this point is a lambda class which
                  prints as <br>
                  >
                  org.wildfly.security.WildflyElytronBaseProvider$$Lambda/0x800000000c<br>
                  > <br>
                  > Several steps further the code enters
                  BootstrapMethodInvoker::invoke <br>
                  > (below the app method call but via 3 hidden
                  frames) with bootstrapMethod <br>
                  > bound to a DirectMethodHandle. After several more
                  steps this enters <br>
                  > DirectMethodHandle$Holder.invokeStatic which in
                  turn calls <br>
                  > MethodHandles::classData(Lookup,String,Class).<br>
                  > <br>
                  > At this point caller is a MethodHandleLookup for
                  the lambda class <br>
                  > Lambda/0x800000000c mentioned above. The
                  following call<br>
                  > <br>
                  >           Object classdata =
                  classData(caller.lookupClass());<br>
                  > <br>
                  > returns null to
                  DirectMethodHandle$Holder.invokeStatic which pops the
                  <br>
                  > same result back out to
                  BootstrapMethodInvoker::invoke at line 90<br>
                  > <br>
                  >                  if (type instanceof
                  Class<?> c) {<br>
                  >                      result =
                  bootstrapMethod.invoke(caller, name, c); <br>
                  > <== null<br>
                  > <br>
                  > This null result pops back out as the value for
                  the call to <br>
                  > BootstrapMethodInvoker.invoke(),
                  ConstantBootstraps.makeConstant() and <br>
                  > MethodHandleNative::linkDynamicConstant().<br>
                  > <br>
                  <br>
                </blockquote>
              </div>
            </blockquote>
          </div>
        </blockquote>
      </div>
    </blockquote>
  </div>

</blockquote></div>