WebKit Bugzilla
New
Browse
Log In
×
Sign in with GitHub
or
Remember my login
Create Account
·
Forgot Password
Forgotten password account recovery
RESOLVED FIXED
76521
WebSocket: MessageEvent fired during send() on workers
https://bugs.webkit.org/show_bug.cgi?id=76521
Summary
WebSocket: MessageEvent fired during send() on workers
Yuta Kitamura
Reported
2012-01-18 00:44:41 PST
Chromium bug:
http://code.google.com/p/chromium/issues/detail?id=108626
There is a bug in WorkerThreadableWebSocketChannel that it may fire a MessageEvent while it is waiting for the result of a synchronous operation (WebSocket.send() or WebSocket.bufferedAmount). The event should not be fired until the user script exits the current cycle of the event loop.
Attachments
Patch
(16.03 KB, patch)
2012-01-18 01:03 PST
,
Yuta Kitamura
no flags
Details
Formatted Diff
Diff
View All
Add attachment
proposed patch, testcase, etc.
Yuta Kitamura
Comment 1
2012-01-18 01:03:06 PST
Created
attachment 122886
[details]
Patch
David Levin
Comment 2
2012-01-18 01:11:58 PST
Comment on
attachment 122886
[details]
Patch View in context:
https://bugs.webkit.org/attachment.cgi?id=122886&action=review
> Source/WebCore/websockets/ThreadableWebSocketChannelClientWrapper.cpp:190 > + if (!m_syncMethodDone) {
Why go through all of this? If you post a task without a mode, then it won't run until the nested loop is done.
Yuta Kitamura
Comment 3
2012-01-18 01:55:23 PST
Comment on
attachment 122886
[details]
Patch View in context:
https://bugs.webkit.org/attachment.cgi?id=122886&action=review
>> Source/WebCore/websockets/ThreadableWebSocketChannelClientWrapper.cpp:190 >> + if (!m_syncMethodDone) { > > Why go through all of this? > > If you post a task without a mode, then it won't run until the nested loop is done.
Sorry, I digged the code a bit but I still don't get the idea of what the "mode" is. WorkerThreadableWebSocketChannel posts tasks with m_taskMode specified as the mode (which is generated within ThreadableWebSocketChannel::create()). Is this wrong?
David Levin
Comment 4
2012-01-18 07:34:37 PST
(In reply to
comment #3
)
> (From update of
attachment 122886
[details]
) > View in context:
https://bugs.webkit.org/attachment.cgi?id=122886&action=review
> > >> Source/WebCore/websockets/ThreadableWebSocketChannelClientWrapper.cpp:190 > >> + if (!m_syncMethodDone) { > > > > Why go through all of this? > > > > If you post a task without a mode, then it won't run until the nested loop is done. > > Sorry, I digged the code a bit but I still don't get the idea of what the "mode" is. > > WorkerThreadableWebSocketChannel posts tasks with m_taskMode specified as the mode (which is generated within ThreadableWebSocketChannel::create()). Is this wrong?
When a nested loop runs in a worker, it runs in a "mode". The mode is just a string but that string acts as a filter. It only allows tasks that are posted using that "mode" (the same string) to run during that nested loop. If a task is posted with no mode, then the task isn't run until the nested loop is done. In general, the code should posts tasks using the m_taskMode (or else they won't run at all during this nested message loop.). However, in this case, for this one task, it sounds like you want to run a task but not until the nested loop is done, so if the code uses "postTask" instead of "postTaskForMode" (or uses "" as the mode if necessary), then you will get the desired result. Hope that helps.
Yuta Kitamura
Comment 5
2012-01-19 02:56:39 PST
Comment on
attachment 122886
[details]
Patch View in context:
https://bugs.webkit.org/attachment.cgi?id=122886&action=review
>>>> Source/WebCore/websockets/ThreadableWebSocketChannelClientWrapper.cpp:190 >>>> + if (!m_syncMethodDone) { >>> >>> Why go through all of this? >>> >>> If you post a task without a mode, then it won't run until the nested loop is done. >> >> Sorry, I digged the code a bit but I still don't get the idea of what the "mode" is. >> >> WorkerThreadableWebSocketChannel posts tasks with m_taskMode specified as the mode (which is generated within ThreadableWebSocketChannel::create()). Is this wrong? > > When a nested loop runs in a worker, it runs in a "mode". The mode is just a string but that string acts as a filter. It only allows tasks that are posted using that "mode" (the same string) to run during that nested loop. If a task is posted with no mode, then the task isn't run until the nested loop is done. > > In general, the code should posts tasks using the m_taskMode (or else they won't run at all during this nested message loop.). However, in this case, for this one task, it sounds like you want to run a task but not until the nested loop is done, so if the code uses "postTask" instead of "postTaskForMode" (or uses "" as the mode if necessary), then you will get the desired result. > > Hope that helps.
Okay, I think I got your point. I'm going to update the patch.
Yuta Kitamura
Comment 6
2012-01-26 19:51:29 PST
Comment on
attachment 122886
[details]
Patch View in context:
https://bugs.webkit.org/attachment.cgi?id=122886&action=review
>>>>> Source/WebCore/websockets/ThreadableWebSocketChannelClientWrapper.cpp:190 >>>>> + if (!m_syncMethodDone) { >>>> >>>> Why go through all of this? >>>> >>>> If you post a task without a mode, then it won't run until the nested loop is done. >>> >>> Sorry, I digged the code a bit but I still don't get the idea of what the "mode" is. >>> >>> WorkerThreadableWebSocketChannel posts tasks with m_taskMode specified as the mode (which is generated within ThreadableWebSocketChannel::create()). Is this wrong? >> >> When a nested loop runs in a worker, it runs in a "mode". The mode is just a string but that string acts as a filter. It only allows tasks that are posted using that "mode" (the same string) to run during that nested loop. If a task is posted with no mode, then the task isn't run until the nested loop is done. >> >> In general, the code should posts tasks using the m_taskMode (or else they won't run at all during this nested message loop.). However, in this case, for this one task, it sounds like you want to run a task but not until the nested loop is done, so if the code uses "postTask" instead of "postTaskForMode" (or uses "" as the mode if necessary), then you will get the desired result. >> >> Hope that helps. > > Okay, I think I got your point. I'm going to update the patch.
After some investigation, I started to think postTask'ing pending tasks (e.g. did*Callback()) is infeasible because we can't call these callbacks when the worker is suspended. We need to process these events in the chronological order, and if we postTask'ed each event it would be difficult to handle suspension correctly. That's why there's m_pendingTasks, I guess. (Maybe the responsibility of suspension handling should exist in WebSocket class, not here; but fixing it would require another work.)
David Levin
Comment 7
2012-01-26 20:38:18 PST
Comment on
attachment 122886
[details]
Patch View in context:
https://bugs.webkit.org/attachment.cgi?id=122886&action=review
Thanks for the explanation. (I think I led you astray.)
> Source/WebCore/websockets/ThreadableWebSocketChannelClientWrapper.cpp:188 > + if (m_suspended)
Since you are adding this check, perhaps in another patch (not this patch), you could change the many places that do this if (!m_suspended) processPendingTasks(); to processPendingTasks();
Yuta Kitamura
Comment 8
2012-02-15 00:19:55 PST
Thanks. I somehow missed you r+ed my patch (sorry!), so I'm trying CQ now...
WebKit Review Bot
Comment 9
2012-02-15 00:44:06 PST
Comment on
attachment 122886
[details]
Patch Clearing flags on attachment: 122886 Committed
r107788
: <
http://trac.webkit.org/changeset/107788
>
WebKit Review Bot
Comment 10
2012-02-15 00:44:12 PST
All reviewed patches have been landed. Closing bug.
Note
You need to
log in
before you can comment on or make changes to this bug.
Top of Page
Format For Printing
XML
Clone This Bug