[icedtea-web] RFC: Patch to use wait() when waiting for conditions to become true
Deepak Bhole
dbhole at redhat.com
Thu Apr 14 13:58:56 PDT 2011
* Dr Andrew John Hughes <ahughes at redhat.com> [2011-04-14 16:32]:
> On 09:22 Thu 14 Apr , Deepak Bhole wrote:
> > * Dr Andrew John Hughes <ahughes at redhat.com> [2011-04-14 07:41]:
> > > On 01:56 Thu 14 Apr , Deepak Bhole wrote:
> > > > * Deepak Bhole <dbhole at redhat.com> [2011-04-13 20:48]:
> > > > > Hi,
> > > > >
> > > > > Andrew mentioned in one of the previous threads that the current plugin
> > > > > code waits in an inefficient manner. It wakes up periodically to
> > > > > check for conditions, and goes back to sleep, and keeps doing that
> > > > > cyclically.
> > > > >
> > > > > The attached patch uses a new function that uses object.wait for
> > > > > the object of interest.
> > > > >
> > > >
> > > > Please see attached new patch. After leaving the office I realized that
> > > > there is a potential race condition in the old patch with locking the
> > > > object inside the wait function. Patch updated to do the locking
> > > > outside (and function comment updated to clearly state that).
> > > >
> > >
> > > Can you elaborate? If it's the same thread acquiring the lock in both
> > > cases, then it should be fine as they are re-entrant i.e. if the function
> > > attempts to lock it and the thread already holds the lock, it's a no-op.
> > >
> >
> > Before, the locking was inside the function, like so:
> >
> > while !<check contition on OBJ>
> > waitTillTimeout(OBJ, timeout)
> > ...
> > ...
> >
> >
> > function ... waitTillTimeout(OBJ, timeout) {
> >
> > lock (OBJ)
> > OBJ.wait()
> > ...
> > }
> >
> > The problem was that condition on OBJ could've been false at check, but then
> > become true before waitTillTimeout() was called. This would result in an
> > incorrect wait until timeout. By moving the lock to before condition checking,
> > the code guarantees that waiting is done only if needed.
> >
>
> Oh yes, that wouldn't work. As I said, we need closures to do that all in one function :-(
>
> > As I wrote this, I realized that in order for the above to work, we would have
> > to lock the hashmap on update too since CHM's locking works differently and may
> > not lock the full table. I will post a separate patch in a bit with the change.
> > In the mean time, I have replied to other concerns below.
> >
>
> As I suggested in the previous mail, we should probably switch back to HMs and do
> all locking in the one place.
>
> > > > Cheers,
> > > > Deepak
> > >
> > > Comments inline. As Jiri said, a ChangeLog would be good too.
> > >
> >
> > Oops, sorry I should've mentioned, the ChangeLog is the same as before
> > (only the locking area changed). Here it is:
> >
> > 2011-04-13 Deepak Bhole <dbhole at redhat.com>
> >
> > * plugin/icedteanp/java/sun/applet/PluginAppletViewer.java (createPanel):
> > use Object.wait() to wait, rather than pariodic sleep.
> > (framePanel): Notify threads waiting on the applets map instance.
> > (appletStateChanged): Notify all threads waiting on the panel that just
> > changed state.
> > (handleMessage): Use the new waitTillTimeout function to wait, rather than
> > periodically waking up.
> > (updateStatus): Notify all threads waiting on status map.
> > (waitForAppletInit): Use the new waitTillTimeout function to wait, rather
> > than periodically waking up.
> > (waitTillTimeout): New function. For a given non-null object, waits until
> > the specified timeout, or, if an interrupt was thrown during wait, returns
> > immediately.
>
> Ok, didn't even read the previous e-mail when I saw it had been superceded.
>
> >
> > > > - // just wait
> > > > + // Wait implemented the long way so that
> > > > + // PluginAppletViewer.waitTillTimeout() needn't be exposed.
> > > > + long maxTimeToSleep = PluginAppletViewer.APPLET_TIMEOUT;
> > > > +
> > > > + synchronized(panel) {
> > > > + while (!panel.isAlive() && maxTimeToSleep > 0) {
> > > > + long sleepStart = System.currentTimeMillis();
> > > > +
> > > > + try {
> > > > + panel.wait(maxTimeToSleep);
> > > > + } catch (InterruptedException e) {} // Just loop back
> > > > +
> > > > + maxTimeToSleep -= (System.currentTimeMillis() - sleepStart);
> > >
> > > So this one matches src.wait() below? It's a little unclear due to the differing
> > > variable names.
> > >
> > > Is there a reason why this doesn't use waitTillTimeout?
> > >
> >
> > I mentioned it in the comments, it is so that
> > PluginAppletViewer.waitTillTimeout() could remain private (the above is in a
> > separate class). I intend to re-factor a lot of that code (already started
> > working on it yesterday) that will address it there. In the mean time, I
> > saw no harm in duplicating a few lines of code.
>
> Oh ok, wasn't clear from just the patch.
>
> >
> > > > }
> > > > }
> > > >
> > > > @@ -307,6 +310,10 @@
> > > > panel.addAppletListener(appletFrame.appletEventListener);
> > > >
> > > > applets.put(identifier, appletFrame);
> > > > +
> > > > + synchronized(applets) {
> > > > + applets.notifyAll();
> > > > + }
> > > >
> > > > PluginDebug.debug(panel, " framed");
> > > > }
> > >
> > > This change is correct, but I'm a bit dubious about having one set of
> > > locks in the CHM for the put and our own lock on applets here. Maybe
> > > revert back to HashMaps and handle all locking in one place?
> > >
> >
> > We would then have to lock on all gets too. CHMs locking is more
> > efficient as it locks on only parts of the table. That is why I chose to
> > do locking only around code that needs it (the API requires wait and
> > notifyall to be calls from threads that own the monitor lock).
>
> CHMs aren't locked on gets either. They do have visibility guarantees.
>
> I'm a little worried about using the monitor of an object that we know does its
> own internal locking. If we're keeping CHMs, we should use our own locks in this
> class so we don't interfere. It doesn't matter what objects are used to synchronise
> on, just that we are consistent.
>
Denis brought this up too. Maybe I am missing something -- even if CHM
does its own locking, wouldn't the caller thread still own the monitor,
thus making it a non-issue? We'd just be locking it earlier. If outside
locking is a concern, that would make CHM very dangerous to use in any
sort of multi-threaded environment that might lock it before an op.
If the notifyAll is a concern -- all docs are very adamant about always
doing wait in a loop as interrupts can be thrown any time, so in worst
case the waiting thread would just wake up when it doesn't need to.
Cheers,
Deepak
> >
>
> I think it would be preferable to make the code clearer. It also removes the confusion with
> src.wait above and any possible conflict with CHM locking. Easier enough to use just an
> Object or the Lock/Condition objects provided by java.util.concurrent.
> > >
> > > Would have been nice to have closures for this. Then you could pass the condition
> > > to this method, and have the whole block handled here. :-)
> > >
> >
> > Only 20 more months if all goes well! :)
> >
> > Cheers,
> > Deepak
> >
> > > > }
> > >
> > >
> > > --
> > > Andrew :)
> > >
> > > Free Java Software Engineer
> > > Red Hat, Inc. (http://www.redhat.com)
> > >
> > > Support Free Java!
> > > Contribute to GNU Classpath and IcedTea
> > > http://www.gnu.org/software/classpath
> > > http://icedtea.classpath.org
> > > PGP Key: F5862A37 (https://keys.indymedia.org/)
> > > Fingerprint = EA30 D855 D50F 90CD F54D 0698 0713 C3ED F586 2A37
>
> --
> Andrew :)
>
> Free Java Software Engineer
> Red Hat, Inc. (http://www.redhat.com)
>
> Support Free Java!
> Contribute to GNU Classpath and IcedTea
> http://www.gnu.org/software/classpath
> http://icedtea.classpath.org
> PGP Key: F5862A37 (https://keys.indymedia.org/)
> Fingerprint = EA30 D855 D50F 90CD F54D 0698 0713 C3ED F586 2A37
More information about the distro-pkg-dev
mailing list