/* * Copyright (C) 2011 Google Inc. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are * met: * * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above * copyright notice, this list of conditions and the following disclaimer * in the documentation and/or other materials provided with the * distribution. * * Neither the name of Google Inc. nor the names of its * contributors may be used to endorse or promote products derived from * this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #ifndef WorkerThreadableWebSocketChannel_h #define WorkerThreadableWebSocketChannel_h #include "core/frame/ConsoleTypes.h" #include "core/workers/WorkerGlobalScope.h" #include "modules/websockets/WebSocketChannel.h" #include "modules/websockets/WebSocketChannelClient.h" #include "wtf/PassOwnPtr.h" #include "wtf/PassRefPtr.h" #include "wtf/RefCounted.h" #include "wtf/RefPtr.h" #include "wtf/Threading.h" #include "wtf/Vector.h" #include "wtf/text/WTFString.h" namespace WebCore { class BlobDataHandle; class KURL; class ExecutionContext; class ThreadableWebSocketChannelClientWrapper; class WorkerGlobalScope; class WorkerLoaderProxy; class WorkerRunLoop; class WorkerThreadableWebSocketChannel : public RefCounted, public WebSocketChannel { WTF_MAKE_FAST_ALLOCATED; public: static PassRefPtr create(WorkerGlobalScope* workerGlobalScope, WebSocketChannelClient* client, const String& taskMode, const String& sourceURL, unsigned lineNumber) { return adoptRef(new WorkerThreadableWebSocketChannel(workerGlobalScope, client, taskMode, sourceURL, lineNumber)); } virtual ~WorkerThreadableWebSocketChannel(); // WebSocketChannel functions. virtual void connect(const KURL&, const String& protocol) OVERRIDE; virtual String subprotocol() OVERRIDE; virtual String extensions() OVERRIDE; virtual WebSocketChannel::SendResult send(const String& message) OVERRIDE; virtual WebSocketChannel::SendResult send(const ArrayBuffer&, unsigned byteOffset, unsigned byteLength) OVERRIDE; virtual WebSocketChannel::SendResult send(PassRefPtr) OVERRIDE; virtual unsigned long bufferedAmount() const OVERRIDE; virtual void close(int code, const String& reason) OVERRIDE; virtual void fail(const String& reason, MessageLevel, const String&, unsigned) OVERRIDE; virtual void disconnect() OVERRIDE; // Will suppress didClose(). virtual void suspend() OVERRIDE; virtual void resume() OVERRIDE; // Generated by the bridge. The Peer and its bridge should have identical // lifetimes. class Peer : public WebSocketChannelClient { WTF_MAKE_NONCOPYABLE(Peer); WTF_MAKE_FAST_ALLOCATED; public: // sourceURLAtConnection and lineNumberAtConnection parameters may // be shown when the connection fails. static Peer* create(PassRefPtr clientWrapper, WorkerLoaderProxy& loaderProxy, ExecutionContext* context, const String& taskMode, const String& sourceURLAtConnection, unsigned lineNumberAtConnection) { return new Peer(clientWrapper, loaderProxy, context, taskMode, sourceURLAtConnection, lineNumberAtConnection); } ~Peer(); void connect(const KURL&, const String& protocol); void send(const String& message); void send(const ArrayBuffer&); void send(PassRefPtr); void bufferedAmount(); void close(int code, const String& reason); void fail(const String& reason, MessageLevel, const String& sourceURL, unsigned lineNumber); void disconnect(); void suspend(); void resume(); // WebSocketChannelClient functions. virtual void didConnect() OVERRIDE; virtual void didReceiveMessage(const String& message) OVERRIDE; virtual void didReceiveBinaryData(PassOwnPtr >) OVERRIDE; virtual void didUpdateBufferedAmount(unsigned long bufferedAmount) OVERRIDE; virtual void didStartClosingHandshake() OVERRIDE; virtual void didClose(unsigned long unhandledBufferedAmount, ClosingHandshakeCompletionStatus, unsigned short code, const String& reason) OVERRIDE; virtual void didReceiveMessageError() OVERRIDE; private: Peer(PassRefPtr, WorkerLoaderProxy&, ExecutionContext*, const String& taskMode, const String& sourceURL, unsigned lineNumber); RefPtr m_workerClientWrapper; WorkerLoaderProxy& m_loaderProxy; RefPtr m_mainWebSocketChannel; String m_taskMode; }; using RefCounted::ref; using RefCounted::deref; protected: // WebSocketChannel functions. virtual void refWebSocketChannel() OVERRIDE { ref(); } virtual void derefWebSocketChannel() OVERRIDE { deref(); } private: // Bridge for Peer. Running on the worker thread. class Bridge : public RefCounted { public: static PassRefPtr create(PassRefPtr workerClientWrapper, PassRefPtr workerGlobalScope, const String& taskMode) { return adoptRef(new Bridge(workerClientWrapper, workerGlobalScope, taskMode)); } ~Bridge(); // sourceURLAtConnection and lineNumberAtConnection parameters may // be shown when the connection fails. void initialize(const String& sourceURLAtConnection, unsigned lineNumberAtConnection); void connect(const KURL&, const String& protocol); WebSocketChannel::SendResult send(const String& message); WebSocketChannel::SendResult send(const ArrayBuffer&, unsigned byteOffset, unsigned byteLength); WebSocketChannel::SendResult send(PassRefPtr); unsigned long bufferedAmount(); void close(int code, const String& reason); void fail(const String& reason, MessageLevel, const String& sourceURL, unsigned lineNumber); void disconnect(); void suspend(); void resume(); using RefCounted::ref; using RefCounted::deref; private: Bridge(PassRefPtr, PassRefPtr, const String& taskMode); static void setWebSocketChannel(ExecutionContext*, Bridge* thisPtr, Peer*, PassRefPtr); // Executed on the main thread to create a Peer for this bridge. // sourceURL and lineNumber provides the source filename and // the line number information at the connection initiation // respectively. They may be shown when the connection fails. static void mainThreadInitialize(ExecutionContext*, WorkerLoaderProxy*, PassRefPtr, const String& taskMode, const String& sourceURL, unsigned lineNumber); // Executed on the worker context's thread. void clearClientWrapper(); void setMethodNotCompleted(); void waitForMethodCompletion(); RefPtr m_workerClientWrapper; RefPtr m_workerGlobalScope; WorkerLoaderProxy& m_loaderProxy; String m_taskMode; Peer* m_peer; }; WorkerThreadableWebSocketChannel(WorkerGlobalScope*, WebSocketChannelClient*, const String& taskMode, const String& sourceURL, unsigned lineNumber); static void mainThreadConnect(ExecutionContext*, Peer*, const KURL&, const String& protocol); static void mainThreadSend(ExecutionContext*, Peer*, const String& message); static void mainThreadSendArrayBuffer(ExecutionContext*, Peer*, PassOwnPtr >); static void mainThreadSendBlob(ExecutionContext*, Peer*, PassRefPtr); static void mainThreadBufferedAmount(ExecutionContext*, Peer*); static void mainThreadClose(ExecutionContext*, Peer*, int code, const String& reason); static void mainThreadFail(ExecutionContext*, Peer*, const String& reason, MessageLevel, const String& sourceURL, unsigned lineNumber); static void mainThreadDestroy(ExecutionContext*, PassOwnPtr); static void mainThreadSuspend(ExecutionContext*, Peer*); static void mainThreadResume(ExecutionContext*, Peer*); class WorkerGlobalScopeDidInitializeTask; RefPtr m_workerGlobalScope; RefPtr m_workerClientWrapper; RefPtr m_bridge; String m_sourceURLAtConnection; unsigned m_lineNumberAtConnection; }; } // namespace WebCore #endif // WorkerThreadableWebSocketChannel_h