RFC: Netx - Set JNLPClassloader as the classloader for system created threads

Deepak Bhole dbhole at redhat.com
Thu Jul 9 11:45:05 PDT 2009


* Omair Majid <omajid at redhat.com> [2009-07-09 11:58]:
> Hi,
>
> The attached patch makes JNLPClassLoader the context classloader for  
> threads created before the JNLP application starts. This deals with the  
> case where the JNLP application needs a previously created thread (such  
> as the Swing Event Dispatch Thread) to load a resource provided by the  
> application.
>
> Bugs:
> https://bugzilla.redhat.com/show_bug.cgi?id=509301
> http://kenai.com/jira/browse/TRIDENT-1?focusedCommentId=15776&page=com.atlassian.jira.plugin.system.issuetabpanels%3Acomment-tabpanel#action_15776
>
>
> ChangeLog:
> 2009-07-09  Omair Majid <omajid at redhat.com>
>
>   * rt/net/sourceforge/jnlp/Launcher.java
>   (launchApplication): Call setContextClassLoaderForAllThreads.
>   (setContextClassLoaderForAllThreads): New function.
>
> Any comments?
>
> Cheers,
> Omair

> diff -r b4bb02c70835 rt/net/sourceforge/jnlp/Launcher.java
> --- a/rt/net/sourceforge/jnlp/Launcher.java	Tue Jul 07 13:44:35 2009 -0400
> +++ b/rt/net/sourceforge/jnlp/Launcher.java	Thu Jul 09 11:16:04 2009 -0400
> @@ -23,6 +23,8 @@
>  import java.io.File;
>  import java.io.IOException;
>  import java.io.InputStream;
> +import java.lang.management.ManagementFactory;
> +import java.lang.management.ThreadMXBean;
>  import java.lang.reflect.Method;
>  import java.net.URL;
>  import java.util.LinkedList;
> @@ -425,8 +427,7 @@
>              Method main = mainClass.getDeclaredMethod("main", new Class[] {String[].class} );
>              String args[] = file.getApplication().getArguments();
>  
> -            // required to make some apps work right
> -            Thread.currentThread().setContextClassLoader(app.getClassLoader());
> +            setContextClassLoaderForAllThreads(app.getClassLoader());
>  
>              if (splashScreen != null) {
>                  if (splashScreen.isSplashScreenValid()) {
> @@ -447,6 +448,45 @@
>          }
>      }
>  
> +    /**
> +     * Set the classloader as the context classloader for all threads. This is
> +     * required to make some applications work. For example, an application that
> +     * provides a custom Swing LnF may ask the swing thread to load resources
> +     * from their JNLP, which would only work if the Swing thread knows about
> +     * the JNLPClassLoader.
> +     * 
> +     * @param classLoader the classloader to set as the context classloader
> +     */
> +    private void setContextClassLoaderForAllThreads(ClassLoader classLoader) {
> +        ThreadMXBean threadBean = ManagementFactory.getThreadMXBean();
> +        ThreadGroup root;
> +        
> +        root = Thread.currentThread().getThreadGroup();
> +        while (root.getParent() != null) {
> +            root = root.getParent();
> +        }
> +
> +        /* be prepared for change in thread size */
> +        int threadCountGuess = threadBean.getThreadCount();
> +        Thread[] threads;
> +        do {
> +            threadCountGuess = threadCountGuess * 2;
> +            threads = new Thread[threadCountGuess];
> +            root.enumerate(threads, true);
> +        } while (threads[threadCountGuess-1] != null);
> +        
> +        
> +        for (Thread thread: threads) {
> +            if (thread != null) {
> +                if (JNLPRuntime.isDebug()) {
> +                    System.err.println("Setting " + classLoader + " as the classloader for thread " + thread.getName());
> +                }
> +                thread.setContextClassLoader(classLoader);
> +            }
> +        }
> +        
> +    }
> +


Assuming you have tested this, go ahead and apply.

Deepak

>      /** 
>       * Launches a JNLP applet. This method should be called from a
>       * thread in the application's thread group.<p>




More information about the distro-pkg-dev mailing list