[icedtea-web] RFC: Patch to stabilize applet initialization
Andrew Su
asu at redhat.com
Fri Oct 22 11:45:01 PDT 2010
----- "Deepak Bhole" <dbhole at redhat.com> wrote:
> From: "Deepak Bhole" <dbhole at redhat.com>
> To: "Andrew Su" <asu at redhat.com>
> Cc: "IcedTea Distro List" <distro-pkg-dev at openjdk.java.net>
> Sent: Friday, October 22, 2010 2:20:34 PM GMT -05:00 US/Canada Eastern
> Subject: Re: [icedtea-web] RFC: Patch to stabilize applet initialization
>
> * Andrew Su <asu at redhat.com> [2010-10-22 12:57]:
> >
> > ----- "Deepak Bhole" <dbhole at redhat.com> wrote:
> >
> > > From: "Deepak Bhole" <dbhole at redhat.com>
> > > To: "Andrew Su" <asu at redhat.com>
> > > Cc: "IcedTea Distro List" <distro-pkg-dev at openjdk.java.net>
> > > Sent: Friday, October 22, 2010 12:06:24 PM GMT -05:00 US/Canada
> Eastern
> > > Subject: Re: [icedtea-web] RFC: Patch to stabilize applet
> initialization
> > >
> > > * Andrew Su <asu at redhat.com> [2010-10-22 10:35]:
> > > >
> > > > ----- "Deepak Bhole" <dbhole at redhat.com> wrote:
> > > >
> > > > > From: "Deepak Bhole" <dbhole at redhat.com>
> > > > > To: "IcedTea Distro List" <distro-pkg-dev at openjdk.java.net>
> > > > > Sent: Thursday, October 21, 2010 5:41:11 PM GMT -05:00
> US/Canada
> > > Eastern
> > > > > Subject: Re: [icedtea-web] RFC: Patch to stabilize applet
> > > initialization
> > > > >
> > > > > * Deepak Bhole <dbhole at redhat.com> [2010-10-21 17:38]:
> > > > > > Hi,
> > > > > >
> > > > > > The attached patch envelopes fixes for two of Andrew's
> pending
> > > > > patches
> > > > > > (fixing 100% CPU load and the frame pop-up issue). In
> addition
> > > to
> > > > > those,
> > > > > > it fixes a lot of other initialization issues, particularly
> > > when
> > > > > > initializing multiple applets and/or closing other tabs
> with
> > > > > applets
> > > > > > which are being initialized.
> > > > > >
> > > > > > With this patch, I tested over 200 parallel applet inits,
> > > randomly
> > > > > > closing tabs, and everything works correctly.
> > > > > >
> > > > > > One known issue is that if timed correctly, a frame can
> appear
> > > > > outside
> > > > > > for a fraction of a second. It always goes away instantly
> > > though,
> > > > > and I
> > > > > > will be posting a patch to fix that problem in the near
> future.
> > > > > >
> > > > > > Cheers,
> > > > > > Deepak
> > > > >
> > > > > Please ignore the spacing issues. I will be normalizing all
> > > spaces
> > > > > for
> > > > > all of icedtea-web in another commit after all pending
> patches
> > > are
> > > > > in.
> > > > >
> > > > > Cheers,
> > > > > Deepak
> > > > >
> > >
> > > <snip>, <cleaned up spacing for the code block below>
> > >
> > > >>private static synchronized PAV_INIT_STATUS updateStatus(int
> > > identifier, PAV_INIT_STATUS newStatus) {
> > > >>
> > > >> PAV_INIT_STATUS prev = status.get(identifier);
> > > >>
> > > >> // If the status is set
> > > >> if (status.containsKey(identifier)) {
> > > >>
> > > >> // Nothing may override destroyed status
> > > >> if
> > > (status.get(identifier).equals(PAV_INIT_STATUS.DESTROYED)) {
> > > >> return prev;
> > > >> }
> > > >>
> > > >> // If status is inactive, only DESTROYED may override
> it
> > > >> if
> (status.get(identifier).equals(PAV_INIT_STATUS.INACTIVE))
> > > {
> > > >> if (!newStatus.equals(PAV_INIT_STATUS.DESTROYED)) {
> > > >> return prev;
> > > >> }
> > > >> }
> > > >> }
> > > >
> > > > Status isn't updated to PAV_INIT_STATUS.DESTROYED.
> > >
> > > Hmm, what do you mean? It is in the line below..
> > I realized, misread the line. :)
> >
> > >
> > > >>
> > > >> // Else set to given status
> > > >> status.put(identifier, newStatus);
> > > >>
> > > >> return prev;
> > > >>}
> > > <snip>
> > >
> > > >
> > > > I believe that splitting it to atleast two queue, one for
> priority
> > > and one for all other messages would be a bit better performance
> wise.
> > >
> > > > Since:
> > > > -- getPriorityStrIfPriority(): assume constant
> > > > -- Adding to proper queue: constant
> > > > -- Checking if queue is empty: constant
> > > > -- Reading and deleting message: constant.
> > > > Right now, doing N checks for each time we try to read a
> message
> > > would be a bit more expensive.
> > > >
> > > > Splitting it up could potentially save us one extra call to
> > > getPriorityStrIfPriority() for each message.
> > > > Using an extra queue, let's call it PQ, we can store the
> priorityStr
> > > into PQ if we determined it is a priority message, this way we
> don't
> > > need to call getPriorityStrIfPriority() on each message again
> later.
> > > >
> > >
> > > I understand and agree with your concern of scanning each message
> > > multiple times, I don't like it either. I don't think multiple
> queues
> > > is
> > > the way to go though. IMO a better solution is to have readQueue
> store
> > > a
> > > structure that stores if a message is priority or not.
> > >
> > > Let's say 5 messages come in. They require 5 scans with the 2
> queue
> > > model to do the placement.
> > >
> > > With a structure-in-queue model, we have:
> > >
> > > Process each message as non-priority. First 2 messages go away.
> Then
> > > we
> > > run bringPriorityMessagesToFront, which scans the remaining 3,
> and
> > > the
> > > priority handlers take care of them.
> > >
> > > The above model becomes inefficient if the queue size is too
> large.
> > > However that is extremely unlikely given that the queue only fills
> up
> > > during multiple-applet init, which is rare. During an average
> run,
> > > queue
> > > size is 0-2.
> > >
> > > What do you think of the above approach?
> > >
> > The above method still requires going through the list of messages.
> As a compromise, I propose that...
> > We keep a structure in readQueue to keep track if it is priority or
> not.
> > With that implemented we can simply do the check when queuing: Place
> message to front if priority and append otherwise.
> >
>
> Hmm, I think there is some confusion here with the costs involved.
> Scanning a message is significantly more expensive than scanning the
> queue. If a message is length 30 (usually a lot more), the cost of
> scanning that is 30. If a queue is size 5, the cost of scanning that
> is
> <items not already scanned>*30 + 5, + 4 (when one thread finishes and
> takes an item) + 3 (when next one finishes and takes an item) + ....
>
> We cannot not go through the messages themselves. To determine if it
> is
> priority, a message _has_ to be scanned. The only difference is do we
> want to scan each one (multiple-queue approach or
> sorting approach as you suggested) or only when needed (single
> queue).
Agreed, all the messages must be scanned at least once, and ideally only once (Which is what we're trying to do). <I wanted to minimized the little bits of constants too :)> But as long as one of the above mentioned methods are implements, I'd give it a thumbs up.
>
> Once a message is scanned, we can guarantee that it never scanned
> again. Thus:
>
> Cost with scanning each msg = C1;
> C1 = totalMsgs*30
>
> Cost of scanning when needed = C2;
> C2 = queueSize*30 + queueSize + queueSize - 1 + ...
> => C2 = queueSize*30 + queueSize^2 (for simplicity);
>
> Therefore,
> C2 > C1 if 1+(queueSize^2)/30 > totalMsgs
>
> In all observed behavior, queueSize very rarely exceeds 5, and even
> if
> it does, it doesn't stay there nearly long enough to catch up with
> the
> total cost of scanning each msg.
>
> > This would remove the need to scan the list of messages and removes
> the need for multiple queue. It'll be sorted already
> > As for the structure used in the queue, it could be simply storing
> the priorityStr, or null.
> >
>
> I am not sure how this differs from my earlier proposal of a
> structure
> that stores whether or not a message in the queue is priority or not..
> I
> was suggesting:
>
> readQueue = [<msg1, isPriority>, <msg2, isPriority> ...]
>
> Aren't we thinking the same thing? :)
If you meant isPriority as the String returned by getPriorityStrIfPriority() then, yes exactly the same!
Cheers,
Andrew
>
> Cheers,
> Deepak
>
> > It's true on average we won't be getting an overwhelming amount of
> messages, but just to cover all our bases :)
> >
> > So, how about this idea?
> >
> > Cheers,
> > Andrew
> >
> >
> > > Cheers,
> > > Deepak
> > >
> > > >
> > > > > > +
> > > > > > public void run() {
> > > > > >
> > > > > > while (true) {
> > > > > > @@ -190,7 +219,6 @@
> > > > > >
> > > > > > String[] msgParts = message.split(" ");
> > > > > >
> > > > > > -
> > > > > > String priorityStr =
> > > > > getPriorityStrIfPriority(message);
> > > > > > boolean isPriorityResponse = (priorityStr
> !=
> > > > > null);
> > > > > >
> > > > > > @@ -199,9 +227,12 @@
> > > > > >
> > > > > > if (worker == null) {
> > > > > > synchronized(readQueue) {
> > > > > > - readQueue.addLast(message);
> > > > > > + readQueue.addFirst(message);
> > > > > > }
> > > > > >
> > > > > > + // re-scan to see if any priority
> message
> > > came
> > > > > in
> > > > > > + bringPriorityMessagesToFront();
> > > > > > +
> > > > > > continue; // re-loop to try next msg
> > > > > > }
> > > > > >
> > > >
> > > > Just the one change to updateStatus.
> > > > The extra queue idea gives a minor boost for performance. (This
> > > change is optional, and can go in later if needed)
> > > > Everything else looks fine. Okay to commit after the change.
> > > >
> > > > Cheers,
> > > > Andrew
More information about the distro-pkg-dev
mailing list