Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(78)

Side by Side Diff: third_party/WebKit/Source/modules/websockets/DocumentWebSocketChannel.cpp

Issue 1574213003: [ABANDONED] Use new Websocket SendBlob IPC from renderer. (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@websocket_blob_send_sender
Patch Set: Minor comment clarifications. Created 4 years, 10 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
1 /* 1 /*
2 * Copyright (C) 2013 Google Inc. All rights reserved. 2 * Copyright (C) 2013 Google Inc. All rights reserved.
3 * 3 *
4 * Redistribution and use in source and binary forms, with or without 4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions are 5 * modification, are permitted provided that the following conditions are
6 * met: 6 * met:
7 * 7 *
8 * * Redistributions of source code must retain the above copyright 8 * * Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer. 9 * notice, this list of conditions and the following disclaimer.
10 * * Redistributions in binary form must reproduce the above 10 * * Redistributions in binary form must reproduce the above
(...skipping 16 matching lines...) Expand all
27 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 27 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
28 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 28 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29 */ 29 */
30 30
31 #include "modules/websockets/DocumentWebSocketChannel.h" 31 #include "modules/websockets/DocumentWebSocketChannel.h"
32 32
33 #include "core/dom/DOMArrayBuffer.h" 33 #include "core/dom/DOMArrayBuffer.h"
34 #include "core/dom/Document.h" 34 #include "core/dom/Document.h"
35 #include "core/dom/ExecutionContext.h" 35 #include "core/dom/ExecutionContext.h"
36 #include "core/fetch/UniqueIdentifier.h" 36 #include "core/fetch/UniqueIdentifier.h"
37 #include "core/fileapi/FileReaderLoader.h"
38 #include "core/fileapi/FileReaderLoaderClient.h"
39 #include "core/frame/LocalFrame.h" 37 #include "core/frame/LocalFrame.h"
40 #include "core/inspector/ConsoleMessage.h" 38 #include "core/inspector/ConsoleMessage.h"
41 #include "core/inspector/InspectorInstrumentation.h" 39 #include "core/inspector/InspectorInstrumentation.h"
42 #include "core/loader/FrameLoader.h" 40 #include "core/loader/FrameLoader.h"
43 #include "core/loader/FrameLoaderClient.h" 41 #include "core/loader/FrameLoaderClient.h"
44 #include "core/loader/MixedContentChecker.h" 42 #include "core/loader/MixedContentChecker.h"
45 #include "modules/websockets/InspectorWebSocketEvents.h" 43 #include "modules/websockets/InspectorWebSocketEvents.h"
46 #include "modules/websockets/WebSocketChannelClient.h" 44 #include "modules/websockets/WebSocketChannelClient.h"
47 #include "modules/websockets/WebSocketFrame.h" 45 #include "modules/websockets/WebSocketFrame.h"
48 #include "platform/Logging.h" 46 #include "platform/Logging.h"
49 #include "platform/network/WebSocketHandshakeRequest.h" 47 #include "platform/network/WebSocketHandshakeRequest.h"
50 #include "platform/weborigin/SecurityOrigin.h" 48 #include "platform/weborigin/SecurityOrigin.h"
51 #include "public/platform/Platform.h" 49 #include "public/platform/Platform.h"
52 #include "public/platform/WebSecurityOrigin.h" 50 #include "public/platform/WebSecurityOrigin.h"
53 #include "public/platform/WebString.h" 51 #include "public/platform/WebString.h"
54 #include "public/platform/WebURL.h" 52 #include "public/platform/WebURL.h"
55 #include "public/platform/WebVector.h" 53 #include "public/platform/WebVector.h"
56 #include "public/platform/modules/websockets/WebSocketHandshakeRequestInfo.h" 54 #include "public/platform/modules/websockets/WebSocketHandshakeRequestInfo.h"
57 #include "public/platform/modules/websockets/WebSocketHandshakeResponseInfo.h" 55 #include "public/platform/modules/websockets/WebSocketHandshakeResponseInfo.h"
58 56
59 using blink::WebSocketHandle; 57 using blink::WebSocketHandle;
60 58
61 namespace blink { 59 namespace blink {
62 60
63 class DocumentWebSocketChannel::BlobLoader final : public GarbageCollectedFinali zed<DocumentWebSocketChannel::BlobLoader>, public FileReaderLoaderClient {
64 public:
65 BlobLoader(PassRefPtr<BlobDataHandle>, DocumentWebSocketChannel*);
66 ~BlobLoader() override { }
67
68 void cancel();
69
70 // FileReaderLoaderClient functions.
71 void didStartLoading() override { }
72 void didReceiveData() override { }
73 void didFinishLoading() override;
74 void didFail(FileError::ErrorCode) override;
75
76 DEFINE_INLINE_TRACE()
77 {
78 visitor->trace(m_channel);
79 }
80
81 private:
82 Member<DocumentWebSocketChannel> m_channel;
83 FileReaderLoader m_loader;
84 };
85
86 DocumentWebSocketChannel::BlobLoader::BlobLoader(PassRefPtr<BlobDataHandle> blob DataHandle, DocumentWebSocketChannel* channel)
87 : m_channel(channel)
88 , m_loader(FileReaderLoader::ReadAsArrayBuffer, this)
89 {
90 m_loader.start(channel->executionContext(), blobDataHandle);
91 }
92
93 void DocumentWebSocketChannel::BlobLoader::cancel()
94 {
95 m_loader.cancel();
96 // didFail will be called immediately.
97 // |this| is deleted here.
98 }
99
100 void DocumentWebSocketChannel::BlobLoader::didFinishLoading()
101 {
102 m_channel->didFinishLoadingBlob(m_loader.arrayBufferResult());
103 // |this| is deleted here.
104 }
105
106 void DocumentWebSocketChannel::BlobLoader::didFail(FileError::ErrorCode errorCod e)
107 {
108 m_channel->didFailLoadingBlob(errorCode);
109 // |this| is deleted here.
110 }
111
112 DocumentWebSocketChannel::DocumentWebSocketChannel(Document* document, WebSocket ChannelClient* client, const String& sourceURL, unsigned lineNumber, WebSocketHa ndle *handle) 61 DocumentWebSocketChannel::DocumentWebSocketChannel(Document* document, WebSocket ChannelClient* client, const String& sourceURL, unsigned lineNumber, WebSocketHa ndle *handle)
113 : ContextLifecycleObserver(document) 62 : ContextLifecycleObserver(document)
114 , m_handle(adoptPtr(handle ? handle : Platform::current()->createWebSocketHa ndle())) 63 , m_handle(adoptPtr(handle ? handle : Platform::current()->createWebSocketHa ndle()))
115 , m_client(client) 64 , m_client(client)
116 , m_identifier(createUniqueIdentifier()) 65 , m_identifier(createUniqueIdentifier())
66 , m_blobLoadingMode(false)
67 , m_blobDataPending(0)
117 , m_sendingQuota(0) 68 , m_sendingQuota(0)
118 , m_receivedDataSizeForFlowControl(receivedDataSizeForFlowControlHighWaterMa rk * 2) // initial quota 69 , m_receivedDataSizeForFlowControl(receivedDataSizeForFlowControlHighWaterMa rk * 2) // initial quota
119 , m_sentSizeOfTopMessage(0) 70 , m_sentSizeOfTopMessage(0)
120 , m_sourceURLAtConstruction(sourceURL) 71 , m_sourceURLAtConstruction(sourceURL)
121 , m_lineNumberAtConstruction(lineNumber) 72 , m_lineNumberAtConstruction(lineNumber)
122 { 73 {
123 } 74 }
124 75
125 DocumentWebSocketChannel::~DocumentWebSocketChannel() 76 DocumentWebSocketChannel::~DocumentWebSocketChannel()
126 { 77 {
127 ASSERT(!m_blobLoader); 78 ASSERT(!m_blobLoadingMode);
128 } 79 }
129 80
130 bool DocumentWebSocketChannel::connect(const KURL& url, const String& protocol) 81 bool DocumentWebSocketChannel::connect(const KURL& url, const String& protocol)
131 { 82 {
132 WTF_LOG(Network, "DocumentWebSocketChannel %p connect()", this); 83 WTF_LOG(Network, "DocumentWebSocketChannel %p connect()", this);
133 if (!m_handle) 84 if (!m_handle)
134 return false; 85 return false;
135 86
136 if (document()->frame()) { 87 if (document()->frame()) {
137 if (MixedContentChecker::shouldBlockWebSocket(document()->frame(), url)) 88 if (MixedContentChecker::shouldBlockWebSocket(document()->frame(), url))
(...skipping 110 matching lines...) Expand 10 before | Expand all | Expand 10 after
248 // handleDidClose may delete this object. 199 // handleDidClose may delete this object.
249 } 200 }
250 201
251 void DocumentWebSocketChannel::disconnect() 202 void DocumentWebSocketChannel::disconnect()
252 { 203 {
253 WTF_LOG(Network, "DocumentWebSocketChannel %p disconnect()", this); 204 WTF_LOG(Network, "DocumentWebSocketChannel %p disconnect()", this);
254 if (m_identifier) { 205 if (m_identifier) {
255 TRACE_EVENT_INSTANT1("devtools.timeline", "WebSocketDestroy", TRACE_EVEN T_SCOPE_THREAD, "data", InspectorWebSocketEvent::data(document(), m_identifier)) ; 206 TRACE_EVENT_INSTANT1("devtools.timeline", "WebSocketDestroy", TRACE_EVEN T_SCOPE_THREAD, "data", InspectorWebSocketEvent::data(document(), m_identifier)) ;
256 InspectorInstrumentation::didCloseWebSocket(document(), m_identifier); 207 InspectorInstrumentation::didCloseWebSocket(document(), m_identifier);
257 } 208 }
258 abortAsyncOperations(); 209 m_blobLoadingMode = false;
259 m_handle.clear(); 210 m_handle.clear();
260 m_client = nullptr; 211 m_client = nullptr;
261 m_identifier = 0; 212 m_identifier = 0;
262 } 213 }
263 214
264 DocumentWebSocketChannel::Message::Message(const CString& text) 215 DocumentWebSocketChannel::Message::Message(const CString& text)
265 : type(MessageTypeText) 216 : type(MessageTypeText)
266 , text(text) { } 217 , text(text) { }
267 218
268 DocumentWebSocketChannel::Message::Message(PassRefPtr<BlobDataHandle> blobDataHa ndle) 219 DocumentWebSocketChannel::Message::Message(PassRefPtr<BlobDataHandle> blobDataHa ndle)
(...skipping 32 matching lines...) Expand 10 before | Expand all | Expand 10 after
301 m_sentSizeOfTopMessage += size; 252 m_sentSizeOfTopMessage += size;
302 m_sendingQuota -= size; 253 m_sendingQuota -= size;
303 *consumedBufferedAmount += size; 254 *consumedBufferedAmount += size;
304 255
305 if (final) { 256 if (final) {
306 m_messages.removeFirst(); 257 m_messages.removeFirst();
307 m_sentSizeOfTopMessage = 0; 258 m_sentSizeOfTopMessage = 0;
308 } 259 }
309 } 260 }
310 261
262 void DocumentWebSocketChannel::sendBlob(const BlobDataHandle& blob)
263 {
264 ASSERT(!m_blobLoadingMode);
265 m_blobLoadingMode = true;
266 m_blobDataPending = blob.size();
267 m_handle->sendBlob(blob.uuid(), blob.size());
268 // The message is still in |m_messages| so that the renderer holds a
269 // reference to it until it is fully sent.
270 }
271
311 void DocumentWebSocketChannel::processSendQueue() 272 void DocumentWebSocketChannel::processSendQueue()
312 { 273 {
313 ASSERT(m_handle); 274 ASSERT(m_handle);
314 uint64_t consumedBufferedAmount = 0; 275 uint64_t consumedBufferedAmount = 0;
315 while (!m_messages.isEmpty() && !m_blobLoader) { 276 while (!m_messages.isEmpty() && !m_blobLoadingMode) {
316 Message* message = m_messages.first().get(); 277 Message* message = m_messages.first().get();
317 if (m_sendingQuota == 0 && message->type != MessageTypeClose) 278 if (m_sendingQuota == 0 && message->type != MessageTypeClose)
318 break; 279 break;
319 switch (message->type) { 280 switch (message->type) {
320 case MessageTypeText: 281 case MessageTypeText:
321 sendInternal(WebSocketHandle::MessageTypeText, message->text.data(), message->text.length(), &consumedBufferedAmount); 282 sendInternal(WebSocketHandle::MessageTypeText, message->text.data(), message->text.length(), &consumedBufferedAmount);
322 break; 283 break;
323 case MessageTypeBlob: 284 case MessageTypeBlob:
324 ASSERT(!m_blobLoader); 285 sendBlob(*message->blobDataHandle);
325 m_blobLoader = new BlobLoader(message->blobDataHandle, this);
326 break; 286 break;
327 case MessageTypeArrayBuffer: 287 case MessageTypeArrayBuffer:
328 sendInternal(WebSocketHandle::MessageTypeBinary, static_cast<const c har*>(message->arrayBuffer->data()), message->arrayBuffer->byteLength(), &consum edBufferedAmount); 288 sendInternal(WebSocketHandle::MessageTypeBinary, static_cast<const c har*>(message->arrayBuffer->data()), message->arrayBuffer->byteLength(), &consum edBufferedAmount);
329 break; 289 break;
330 case MessageTypeTextAsCharVector: 290 case MessageTypeTextAsCharVector:
331 sendInternal(WebSocketHandle::MessageTypeText, message->vectorData-> data(), message->vectorData->size(), &consumedBufferedAmount); 291 sendInternal(WebSocketHandle::MessageTypeText, message->vectorData-> data(), message->vectorData->size(), &consumedBufferedAmount);
332 break; 292 break;
333 case MessageTypeBinaryAsCharVector: 293 case MessageTypeBinaryAsCharVector:
334 sendInternal(WebSocketHandle::MessageTypeBinary, message->vectorData ->data(), message->vectorData->size(), &consumedBufferedAmount); 294 sendInternal(WebSocketHandle::MessageTypeBinary, message->vectorData ->data(), message->vectorData->size(), &consumedBufferedAmount);
335 break; 295 break;
(...skipping 13 matching lines...) Expand all
349 309
350 void DocumentWebSocketChannel::flowControlIfNecessary() 310 void DocumentWebSocketChannel::flowControlIfNecessary()
351 { 311 {
352 if (!m_handle || m_receivedDataSizeForFlowControl < receivedDataSizeForFlowC ontrolHighWaterMark) { 312 if (!m_handle || m_receivedDataSizeForFlowControl < receivedDataSizeForFlowC ontrolHighWaterMark) {
353 return; 313 return;
354 } 314 }
355 m_handle->flowControl(m_receivedDataSizeForFlowControl); 315 m_handle->flowControl(m_receivedDataSizeForFlowControl);
356 m_receivedDataSizeForFlowControl = 0; 316 m_receivedDataSizeForFlowControl = 0;
357 } 317 }
358 318
359 void DocumentWebSocketChannel::abortAsyncOperations()
360 {
361 if (m_blobLoader) {
362 m_blobLoader->cancel();
363 m_blobLoader.clear();
364 }
365 }
366
367 void DocumentWebSocketChannel::handleDidClose(bool wasClean, unsigned short code , const String& reason) 319 void DocumentWebSocketChannel::handleDidClose(bool wasClean, unsigned short code , const String& reason)
368 { 320 {
369 m_handle.clear(); 321 m_handle.clear();
370 abortAsyncOperations(); 322 m_blobLoadingMode = false;
371 if (!m_client) { 323 if (!m_client) {
372 return; 324 return;
373 } 325 }
374 WebSocketChannelClient* client = m_client; 326 WebSocketChannelClient* client = m_client;
375 m_client = nullptr; 327 m_client = nullptr;
376 WebSocketChannelClient::ClosingHandshakeCompletionStatus status = 328 WebSocketChannelClient::ClosingHandshakeCompletionStatus status =
377 wasClean ? WebSocketChannelClient::ClosingHandshakeComplete : WebSocketC hannelClient::ClosingHandshakeIncomplete; 329 wasClean ? WebSocketChannelClient::ClosingHandshakeComplete : WebSocketC hannelClient::ClosingHandshakeIncomplete;
378 client->didClose(status, code, reason); 330 client->didClose(status, code, reason);
379 // client->didClose may delete this object. 331 // client->didClose may delete this object.
380 } 332 }
(...skipping 127 matching lines...) Expand 10 before | Expand all | Expand 10 after
508 460
509 void DocumentWebSocketChannel::didReceiveFlowControl(WebSocketHandle* handle, in t64_t quota) 461 void DocumentWebSocketChannel::didReceiveFlowControl(WebSocketHandle* handle, in t64_t quota)
510 { 462 {
511 WTF_LOG(Network, "DocumentWebSocketChannel %p didReceiveFlowControl(%p, %ld) ", this, handle, static_cast<long>(quota)); 463 WTF_LOG(Network, "DocumentWebSocketChannel %p didReceiveFlowControl(%p, %ld) ", this, handle, static_cast<long>(quota));
512 464
513 ASSERT(m_handle); 465 ASSERT(m_handle);
514 ASSERT(handle == m_handle); 466 ASSERT(handle == m_handle);
515 ASSERT(quota >= 0); 467 ASSERT(quota >= 0);
516 468
517 m_sendingQuota += quota; 469 m_sendingQuota += quota;
518 processSendQueue(); 470 if (m_blobLoadingMode) {
471 uint64_t blobUsedQuota = std::min(static_cast<uint64_t>(quota), m_blobDa taPending);
472 if (blobUsedQuota > 0) {
473 m_sendingQuota -= blobUsedQuota;
474 m_blobDataPending -= blobUsedQuota;
475 if (m_client)
476 m_client->didConsumeBufferedAmount(blobUsedQuota);
tyoshino (SeeGerritForStatus) 2016/02/09 14:29:10 IIUC, it's possible that this would process flow c
Adam Rice 2016/02/10 01:37:55 Correct. For example, if the user sends an 8KB Tex
tyoshino (SeeGerritForStatus) 2016/02/10 06:48:46 Right. bufferedAmount doesn't tell the actual memo
Adam Rice 2016/02/10 12:58:27 Comment added. I tried to keep it short to match B
477 }
478 } else {
479 processSendQueue();
480 }
519 } 481 }
520 482
521 void DocumentWebSocketChannel::didStartClosingHandshake(WebSocketHandle* handle) 483 void DocumentWebSocketChannel::didStartClosingHandshake(WebSocketHandle* handle)
522 { 484 {
523 WTF_LOG(Network, "DocumentWebSocketChannel %p didStartClosingHandshake(%p)", this, handle); 485 WTF_LOG(Network, "DocumentWebSocketChannel %p didStartClosingHandshake(%p)", this, handle);
524 486
525 ASSERT(m_handle); 487 ASSERT(m_handle);
526 ASSERT(handle == m_handle); 488 ASSERT(handle == m_handle);
527 489
528 if (m_client) 490 if (m_client)
529 m_client->didStartClosingHandshake(); 491 m_client->didStartClosingHandshake();
530 } 492 }
531 493
532 void DocumentWebSocketChannel::didFinishLoadingBlob(PassRefPtr<DOMArrayBuffer> b uffer) 494 void DocumentWebSocketChannel::didCompleteSendingBlob(WebSocketHandle* handle)
533 { 495 {
534 m_blobLoader.clear(); 496 WTF_LOG(Network, "DocumentWebSocketChannel %p didCompleteSendingBlob(%p)", t his, handle);
497
498 // If the IPC was unexpected, do nothing.
499 if (!m_blobLoadingMode)
500 return;
501
535 ASSERT(m_handle); 502 ASSERT(m_handle);
536 // The loaded blob is always placed on m_messages[0]. 503 ASSERT(handle == m_handle);
537 ASSERT(m_messages.size() > 0 && m_messages.first()->type == MessageTypeBlob) ; 504 ASSERT(m_sendingQuota >= m_blobDataPending);
538 // We replace it with the loaded blob. 505
539 m_messages.first() = adoptPtr(new Message(buffer)); 506 m_sendingQuota -= m_blobDataPending;
507 if (m_blobDataPending && m_client)
508 m_client->didConsumeBufferedAmount(m_blobDataPending);
509 m_blobDataPending = 0;
510 m_blobLoadingMode = false;
511 // Remove the reference to the Blob.
512 m_messages.removeFirst();
540 processSendQueue(); 513 processSendQueue();
541 } 514 }
542 515
543 void DocumentWebSocketChannel::didFailLoadingBlob(FileError::ErrorCode errorCode )
544 {
545 m_blobLoader.clear();
546 if (errorCode == FileError::ABORT_ERR) {
547 // The error is caused by cancel().
548 return;
549 }
550 // FIXME: Generate human-friendly reason message.
551 failAsError("Failed to load Blob: error code = " + String::number(errorCode) );
552 // |this| can be deleted here.
553 }
554
555 DEFINE_TRACE(DocumentWebSocketChannel) 516 DEFINE_TRACE(DocumentWebSocketChannel)
556 { 517 {
557 visitor->trace(m_blobLoader);
558 visitor->trace(m_client); 518 visitor->trace(m_client);
559 WebSocketChannel::trace(visitor); 519 WebSocketChannel::trace(visitor);
560 ContextLifecycleObserver::trace(visitor); 520 ContextLifecycleObserver::trace(visitor);
561 } 521 }
562 522
563 } // namespace blink 523 } // namespace blink
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698