premain: Crash in Infinispan server code caused by NPE under MethodHandleNative::linkDynamicConstant

Ashutosh Mehra asmehra at redhat.com
Wed Sep 11 14:14:03 UTC 2024


Hi Andrew,
Thanks for sharing the initial investigation.
I have been looking into this and have a few of things to add to your
analysis:

1.  As you mentioned the classData for the lambda
class WildFlyElytronBaseProvider$$Lambda is null.
The classData is stored in the mirror object of the InstanceKlass when the
class is defined through JVM_LookupDefineClass.
However, when we create the scratch mirror object (which get stored in the
AOT cache) the classData is not populated.
See
https://github.com/openjdk/leyden/blob/d23b9f2d5e3523cc547337da59327ed86a6057a3/src/hotspot/share/classfile/javaClasses.cpp#L1128-L1131

  Handle classData; // set to null. Will be reinitialized at runtime
  Handle mirror;
  Handle comp_mirror;
  allocate_mirror(k, /*is_scratch=*/true, protection_domain, classData,
mirror, comp_mirror, CHECK);

So this explains why the call to classData(caller.lookupClass()) returned
null.

2. In the mainline there is a check in InnerClassLambdaMetafactory.java for
the particular code pattern used by the application.
If this code pattern is found then the lambda proxy class is not included
in the CDS archive.
See
https://github.com/openjdk/leyden/blob/d23b9f2d5e3523cc547337da59327ed86a6057a3/src/java.base/share/classes/java/lang/invoke/InnerClassLambdaMetafactory.java#L163-L170
and
https://github.com/openjdk/leyden/blob/d23b9f2d5e3523cc547337da59327ed86a6057a3/src/java.base/share/classes/java/lang/invoke/InnerClassLambdaMetafactory.java#L246

        // If the target class invokes a protected method inherited from a
        // superclass in a different package, or does 'invokespecial', the
        // lambda class has no access to the resolved method, or does
        // 'invokestatic' on a hidden class which cannot be resolved by
name.
        // Instead, we need to pass the live implementation method handle to
        // the proxy class to invoke directly. (javac prefers to avoid this
        // situation by generating bridges in the target class)
        useImplMethodHandle =
(Modifier.isProtected(implInfo.getModifiers()) &&
                               !VerifyAccess.isSamePackage(targetClass,
implInfo.getDeclaringClass())) ||
                               implKind ==
MethodHandleInfo.REF_invokeSpecial ||
                               implKind ==
MethodHandleInfo.REF_invokeStatic && implClass.isHidden();

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.

3. For the null classData problem mentioned above, I tried to fix it by
storing classData in the scratch mirror using the following patch:

diff --git a/src/hotspot/share/classfile/javaClasses.cpp
b/src/hotspot/share/classfile/javaClasses.cpp
index bd8141adbcc..41766e98093 100644
--- a/src/hotspot/share/classfile/javaClasses.cpp
+++ b/src/hotspot/share/classfile/javaClasses.cpp
@@ -1094,9 +1094,9 @@ void java_lang_Class::create_mirror(Klass* k, Handle
class_loader,
     }
     if (CDSConfig::is_dumping_heap()) {
       if (CDSConfig::is_dumping_protection_domains()) {
-        create_scratch_mirror(k, protection_domain, CHECK);
+        create_scratch_mirror(k, protection_domain, classData, CHECK);
       } else {
-        create_scratch_mirror(k, Handle() /* null protection_domain*/,
CHECK);
+        create_scratch_mirror(k, Handle() /* null protection_domain*/,
classData, CHECK);
       }
     }
   } else {
@@ -1117,7 +1117,7 @@ void java_lang_Class::create_mirror(Klass* k, Handle
class_loader,
 // Note: we archive the "scratch mirror" instead of k->java_mirror(),
because the
 // latter may contain dumptime-specific information that cannot be archived
 // (e.g., ClassLoaderData*, or static fields that are modified by Java
code execution).
-void java_lang_Class::create_scratch_mirror(Klass* k, Handle
protection_domain, TRAPS) {
+void java_lang_Class::create_scratch_mirror(Klass* k, Handle
protection_domain, Handle classData, TRAPS) {
   if (k->class_loader() != nullptr &&
       k->class_loader() != SystemDictionary::java_platform_loader() &&
       k->class_loader() != SystemDictionary::java_system_loader()) {
@@ -1125,9 +1125,11 @@ void java_lang_Class::create_scratch_mirror(Klass*
k, Handle protection_domain,
     return;
   }

-  Handle classData; // set to null. Will be reinitialized at runtime
+  //Handle classData; // set to null. Will be reinitialized at runtime
   Handle mirror;
   Handle comp_mirror;
   allocate_mirror(k, /*is_scratch=*/true, protection_domain, classData,
mirror, comp_mirror, CHECK);

   if (comp_mirror() != nullptr) {
diff --git a/src/hotspot/share/classfile/javaClasses.hpp
b/src/hotspot/share/classfile/javaClasses.hpp
index bc49a0861a7..7ec2a2556dd 100644
--- a/src/hotspot/share/classfile/javaClasses.hpp
+++ b/src/hotspot/share/classfile/javaClasses.hpp
@@ -263,7 +263,7 @@ class java_lang_Class : AllStatic {

   // Archiving
   static void serialize_offsets(SerializeClosure* f) NOT_CDS_RETURN;
-  static void create_scratch_mirror(Klass* k, Handle protection_domain,
TRAPS) NOT_CDS_JAVA_HEAP_RETURN;
+  static void create_scratch_mirror(Klass* k, Handle protection_domain,
Handle classData, TRAPS) NOT_CDS_JAVA_HEAP_RETURN;
   static bool restore_archived_mirror(Klass *k, Handle class_loader,
Handle module,
                                       Handle protection_domain,
                                       TRAPS)
NOT_CDS_JAVA_HEAP_RETURN_(false);

But this resulted in a different exception:

Exception in thread "main" java.lang.ExceptionInInitializerError
at com.redhat.leyden.Main.main(Main.java:7)
Caused by: java.lang.invoke.WrongMethodTypeException: handle's method type
(WildFlyElytronBaseProvider,Service)void but found
(WildFlyElytronBaseProvider,Service)void
at
java.base/java.lang.invoke.Invokers.newWrongMethodTypeException(Invokers.java:521)
at java.base/java.lang.invoke.Invokers.checkExactType(Invokers.java:530)
at
java.base/java.lang.invoke.Invokers$Holder.invokeExact_MT(Invokers$Holder)
at
org.wildfly.security.WildFlyElytronBaseProvider$$Lambda/0x80000000c.accept(Unknown
Source)
at
org.wildfly.security.WildFlyElytronBaseProvider.putMakedPasswordImplementations(WildFlyElytronBaseProvider.java:112)
at
org.wildfly.security.WildFlyElytronBaseProvider.putPasswordImplementations(WildFlyElytronBaseProvider.java:107)
at
org.wildfly.security.password.WildFlyElytronPasswordProvider.<init>(WildFlyElytronPasswordProvider.java:43)
at
org.wildfly.security.password.WildFlyElytronPasswordProvider.<clinit>(WildFlyElytronPasswordProvider.java:36)
... 1 more

The exception message is strange because the handle's method type and the
expected type are both symbolically the same.
I am debugging this exception at the moment.

Thanks,
- Ashutosh Mehra


On Wed, Sep 11, 2024 at 6:03 AM Andrew Dinn <adinn at redhat.com> wrote:

> Oops, sorry, I debugged this a few days ago! Correction to a few details:
>
> On 11/09/2024 10:39, Andrew Dinn wrote:
> > A crash due to an NPE was observed in the Infinispan (Data Grid) server
> > app when deployed using the Leyden EA. The crash still manifests with
> > the latest premain code. The crash happens below an application call
> > which employs a method reference as argument
> >
> >          putMakedPasswordImplementations(this::putService, this);
>
> The called method in turn calls consumer.accept
>
>              consumer.accept(new Service(provider,
> PASSWORD_FACTORY_TYPE, algorithm,
> "org.wildfly.security.password.impl.PasswordFactorySpiImpl", emptyList,
> emptyMap));
>
> which enters enters MethodHandleNative::linkDynamicConstant()
>
> > Debugging shows that the call to linkDynamicConstant() returns null.
> >
> > A simple reproducer for the problem is available as a maven project on
> > github:
> >
> >    https://github.com/tristantarrant/elytron-leyden
> >
> > The ReadMe provides an explanation of how to reproduce the problem. I
> > did so and the debugged to find out some of the details of what is
> > happening (see below) but did not fully clarify the problem. Help from
> > someone more conversant with the ins and outs of method handle
> > bootstraps in premain would be welcome. Details follow.
> >
> > regards,
> >
> >
> > Andrew Dinn
> > -----------
> >
> > I downloaded the git repo and attached the Java sources using Maven
> command
> >
> >    $ mvn dependency:sources
> >
> > Having manifested the crash by following the instructions in the README
> > I reran the leyden JVM under gdb using the following commands to enable
> > Java debugging
> >
> > $ gdb ${LEYDEN_HOME}/bin/java
> > (gdb) cd /path/to/mvn/project
> > (gdb) run
> > -agentlib:jdwp=transport=dt_socket,server=y,suspend=y,address=5005
> > -classpath
> >
> /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/
> 2.5.1.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
> >
> > The problem manifests at WildflyElytronBaseProvider.java:112 in method
> > WildflyElytronBaseProvider::putMakedPasswordImplementations
>
>      static void putMakedPasswordImplementations(Consumer<Service>
> consumer, Provider provider) {
>          for (String algorithm : MASKED_ALGORITHMS) {
>              consumer.accept(new Service(provider,
> PASSWORD_FACTORY_TYPE, algorithm,
> "org.wildfly.security.password.impl.PasswordFactorySpiImpl", emptyList,
> emptyMap)); <== NPE under this call
>          }
>
>
> > The source code for this method can be found in the following source jar
> >
> >
> >
> ${M2_REPO}/org/wildfly/security/wildfly-elytron-base/2.5.1.Final/wildfly-elytron-base-2.5.1.Final-sources.jar
> >
> > (where M2_REPO will normally be ~/.m2/repository)
> >
> > Stepping into accept eventually enters
> MethodHandleNative::linkDynamicConstant
> > which in turn enters into ConstantBootstraps.makeConstant(). The caller
> > Class at this point is a lambda class which prints as
> > org.wildfly.security.WildflyElytronBaseProvider$$Lambda/0x800000000c
> >
> > Several steps further the code enters BootstrapMethodInvoker::invoke
> > (below the app method call but via 3 hidden frames) with bootstrapMethod
> > bound to a DirectMethodHandle. After several more steps this enters
> > DirectMethodHandle$Holder.invokeStatic which in turn calls
> > MethodHandles::classData(Lookup,String,Class).
> >
> > At this point caller is a MethodHandleLookup for the lambda class
> > Lambda/0x800000000c mentioned above. The following call
> >
> >           Object classdata = classData(caller.lookupClass());
> >
> > returns null to DirectMethodHandle$Holder.invokeStatic which pops the
> > same result back out to BootstrapMethodInvoker::invoke at line 90
> >
> >                  if (type instanceof Class<?> c) {
> >                      result = bootstrapMethod.invoke(caller, name, c);
> > <== null
> >
> > This null result pops back out as the value for the call to
> > BootstrapMethodInvoker.invoke(), ConstantBootstraps.makeConstant() and
> > MethodHandleNative::linkDynamicConstant().
> >
>
>
-------------- next part --------------
An HTML attachment was scrubbed...
URL: <https://mail.openjdk.org/pipermail/leyden-dev/attachments/20240911/7b8e339a/attachment.htm>


More information about the leyden-dev mailing list