[rfc][icedtea-web] A dead-lock / Firefox hanging fix

Adam Domurad adomurad at redhat.com
Tue Apr 16 13:15:37 PDT 2013


Hi all.

Recently on one of my computers I noticed that Firefox was hanging every 
time an applet failed. While I'm not sure why this was suddenly 
happening every time, I took it as an opportunity to look into a fix 
while the bug was in plain sight.

The problem area is here:

plugin/icedteanp/java/sun/applet/PluginAppletViewer.java:740 (from HEAD)
>                 while (panel == null || !panel.isAlive()) {
>                                          ^^^^^^^^^^^^^
>                     maxTimeToSleep -= waitTillTimeout(panelLock, 
> panelLive,
> maxTimeToSleep);
>
>                     /* we already waited till timeout, give up here 
> directly,
>                      *  instead of waiting 180s again in below 
> waitForAppletInit()
>                      */
>                     if(maxTimeToSleep < 0) {
>                         streamhandler.write("instance " + identifier + 
> " reference " + -1 + " fatalError: " + "Initialization timed out");
>                         return;

The intent for using panel.isAlive() in the condition here is presumably 
to ensure that the handler thread is created. NetxPanel is created with 
appletAlive=true, so the following method:

netx/net/sourceforge/jnlp/NetxPanel.java:196 (from HEAD)
>     public boolean isAlive() {
>         return handler != null && handler.isAlive() && this.appletAlive;
>     }

Effectively tests that the handler thread was created. There's one 
problem, in the case of an exception path ...

netx/net/sourceforge/jnlp/NetxPanel.java:150 (from HEAD)
>         } catch (Exception e) {
>             this.appletAlive = false;
>                   ^^^^^^^^^^^^^^^^^^
>             e.printStackTrace();
> replaceSplash(SplashUtils.getErrorSplashScreen(getWidth(), 
> getHeight(), e));

This unfortunately causes isAlive() to permanently return false from 
that point onwards. Clearly if isAlive() ever gets false in the 
condition at PluginAppletViewer:740 we will loop forever. The C++ side 
of the plugin can do nothing but wait. Additionally, since icedtea-web 
is run in-process unlike other plugins, this hangs firefox completely.

One thing to notice is, PluginAppletViewer:740-751 are fairly redundant 
because line 760 (== 'waitForAppletInit(panel)') waits for complete 
initialization anyway.
The patch therefore axes these lines, which were mostly identical to 
waitForAppletInit. There is no need to separately wait for the handler 
thread to initialize. It's also worth noting that waiting on panel == 
null makes little sense because 'panel' is set *only* in the 
constructor. Therefore if it were ever null we would be spinning forever 
anyway.

The other part of the fix was making sure the C++ side of the plugin 
gets a response.

plugin/icedteanp/java/sun/applet/PluginAppletViewer.java:764 (from HEAD)
>             if (panel.getApplet() == null) {
>                 streamhandler.write("instance " + identifier + " 
> reference " + -1 + " fatalError: " + "Initialization failed");
>                 return;
>             }

This lacks a response to the message being handled, meaning the C++ side 
will wait until timeout. Adding 'streamhandler.write("context 0 
reference " + reference + " Error");' solves the problem neatly.


ChangeLog:
2013-XX-XX  Adam Domurad  <adomurad at redhat.com>

         Fix a dead-lock that can cause (namely) Firefox to hang.
         * netx/net/sourceforge/jnlp/NetxPanel.java
         (appletAlive): Remove flag.
         (isAlive): Remove getter.
         (initialized): New, explicit initialization flag.
         (isInitialized): New, getter.
         (runLoader): Set initialization flag when done (whether errored 
or not).
         * plugin/icedteanp/java/sun/applet/PluginAppletViewer.java
         (waitForAppletInit): Wait on initialization flag from NetxPanel.
         (handleMessage): Remove redundant waiting for init. Respond 
properly to
         GetJavaObject in case of error/time-out.


Happy hacking,
-Adam
-------------- next part --------------
A non-text attachment was scrubbed...
Name: NetxPanel-init-deadlock.patch
Type: text/x-patch
Size: 5005 bytes
Desc: not available
Url : http://mail.openjdk.java.net/pipermail/distro-pkg-dev/attachments/20130416/0a13b8cc/NetxPanel-init-deadlock.patch 


More information about the distro-pkg-dev mailing list