/* * Copyright (C) 2008 Apple Inc. All Rights Reserved. * Copyright (C) 2009 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: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. 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. * * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``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 APPLE INC. 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. * */ #include "config.h" #include "WorkerMessagingProxy.h" #include "ContentSecurityPolicy.h" #include "DOMWindow.h" #include "DedicatedWorkerGlobalScope.h" #include "DedicatedWorkerThread.h" #include "Document.h" #include "ErrorEvent.h" #include "Event.h" #include "EventNames.h" #include "ExceptionCode.h" #include "MessageEvent.h" #include "PageGroup.h" #include "ScriptExecutionContext.h" #include "Worker.h" #include #include #include namespace WebCore { WorkerGlobalScopeProxy* WorkerGlobalScopeProxy::create(Worker* worker) { return new WorkerMessagingProxy(worker); } WorkerMessagingProxy::WorkerMessagingProxy(Worker* workerObject) : m_scriptExecutionContext(workerObject->scriptExecutionContext()) , m_workerObject(workerObject) , m_mayBeDestroyed(false) , m_unconfirmedMessageCount(0) , m_workerThreadHadPendingActivity(false) , m_askedToTerminate(false) { ASSERT(m_workerObject); ASSERT((is(*m_scriptExecutionContext) && isMainThread()) || (is(*m_scriptExecutionContext) && currentThread() == downcast(*m_scriptExecutionContext).thread().threadID())); } WorkerMessagingProxy::~WorkerMessagingProxy() { ASSERT(!m_workerObject); ASSERT((is(*m_scriptExecutionContext) && isMainThread()) || (is(*m_scriptExecutionContext) && currentThread() == downcast(*m_scriptExecutionContext).thread().threadID())); } void WorkerMessagingProxy::startWorkerGlobalScope(const URL& scriptURL, const String& userAgent, const String& sourceCode, const ContentSecurityPolicyResponseHeaders& contentSecurityPolicyResponseHeaders, bool shouldBypassMainWorldContentSecurityPolicy, WorkerThreadStartMode startMode) { // FIXME: This need to be revisited when we support nested worker one day ASSERT(m_scriptExecutionContext); Document& document = downcast(*m_scriptExecutionContext); RefPtr thread = DedicatedWorkerThread::create(scriptURL, userAgent, sourceCode, *this, *this, startMode, contentSecurityPolicyResponseHeaders, shouldBypassMainWorldContentSecurityPolicy, document.topOrigin()); workerThreadCreated(thread); thread->start(); } void WorkerMessagingProxy::postMessageToWorkerObject(PassRefPtr message, std::unique_ptr channels) { MessagePortChannelArray* channelsPtr = channels.release(); m_scriptExecutionContext->postTask([this, channelsPtr, message] (ScriptExecutionContext& context) { Worker* workerObject = this->workerObject(); if (!workerObject || askedToTerminate()) return; std::unique_ptr ports = MessagePort::entanglePorts(context, std::unique_ptr(channelsPtr)); workerObject->dispatchEvent(MessageEvent::create(WTFMove(ports), message)); }); } void WorkerMessagingProxy::postMessageToWorkerGlobalScope(PassRefPtr message, std::unique_ptr channels) { if (m_askedToTerminate) return; MessagePortChannelArray* channelsPtr = channels.release(); ScriptExecutionContext::Task task([channelsPtr, message] (ScriptExecutionContext& scriptContext) { ASSERT_WITH_SECURITY_IMPLICATION(scriptContext.isWorkerGlobalScope()); DedicatedWorkerGlobalScope& context = static_cast(scriptContext); std::unique_ptr ports = MessagePort::entanglePorts(scriptContext, std::unique_ptr(channelsPtr)); context.dispatchEvent(MessageEvent::create(WTFMove(ports), message)); context.thread().workerObjectProxy().confirmMessageFromWorkerObject(context.hasPendingActivity()); }); if (m_workerThread) { ++m_unconfirmedMessageCount; m_workerThread->runLoop().postTask(WTFMove(task)); } else m_queuedEarlyTasks.append(std::make_unique(WTFMove(task))); } void WorkerMessagingProxy::postTaskToLoader(ScriptExecutionContext::Task task) { // FIXME: In case of nested workers, this should go directly to the root Document context. ASSERT(m_scriptExecutionContext->isDocument()); m_scriptExecutionContext->postTask(WTFMove(task)); } bool WorkerMessagingProxy::postTaskForModeToWorkerGlobalScope(ScriptExecutionContext::Task task, const String& mode) { if (m_askedToTerminate) return false; ASSERT(m_workerThread); m_workerThread->runLoop().postTaskForMode(WTFMove(task), mode); return true; } void WorkerMessagingProxy::postExceptionToWorkerObject(const String& errorMessage, int lineNumber, int columnNumber, const String& sourceURL) { StringCapture capturedErrorMessage(errorMessage); StringCapture capturedSourceURL(sourceURL); m_scriptExecutionContext->postTask([this, capturedErrorMessage, capturedSourceURL, lineNumber, columnNumber] (ScriptExecutionContext& context) { Worker* workerObject = this->workerObject(); if (!workerObject) return; // We don't bother checking the askedToTerminate() flag here, because exceptions should *always* be reported even if the thread is terminated. // This is intentionally different than the behavior in MessageWorkerTask, because terminated workers no longer deliver messages (section 4.6 of the WebWorker spec), but they do report exceptions. bool errorHandled = !workerObject->dispatchEvent(ErrorEvent::create(capturedErrorMessage.string(), capturedSourceURL.string(), lineNumber, columnNumber)); if (!errorHandled) context.reportException(capturedErrorMessage.string(), lineNumber, columnNumber, capturedSourceURL.string(), 0); }); } void WorkerMessagingProxy::postConsoleMessageToWorkerObject(MessageSource source, MessageLevel level, const String& message, int lineNumber, int columnNumber, const String& sourceURL) { StringCapture capturedMessage(message); StringCapture capturedSourceURL(sourceURL); m_scriptExecutionContext->postTask([this, source, level, capturedMessage, capturedSourceURL, lineNumber, columnNumber] (ScriptExecutionContext& context) { if (askedToTerminate()) return; context.addConsoleMessage(source, level, capturedMessage.string(), capturedSourceURL.string(), lineNumber, columnNumber); }); } void WorkerMessagingProxy::workerThreadCreated(PassRefPtr workerThread) { m_workerThread = workerThread; if (m_askedToTerminate) { // Worker.terminate() could be called from JS before the thread was created. m_workerThread->stop(); } else { ASSERT(!m_unconfirmedMessageCount); m_unconfirmedMessageCount = m_queuedEarlyTasks.size(); m_workerThreadHadPendingActivity = true; // Worker initialization means a pending activity. auto queuedEarlyTasks = WTFMove(m_queuedEarlyTasks); for (auto& task : queuedEarlyTasks) m_workerThread->runLoop().postTask(WTFMove(*task)); } } void WorkerMessagingProxy::workerObjectDestroyed() { m_workerObject = 0; m_scriptExecutionContext->postTask([this] (ScriptExecutionContext&) { m_mayBeDestroyed = true; if (m_workerThread) terminateWorkerGlobalScope(); else workerGlobalScopeDestroyedInternal(); }); } void WorkerMessagingProxy::notifyNetworkStateChange(bool isOnline) { if (m_askedToTerminate) return; if (!m_workerThread) return; m_workerThread->runLoop().postTask([isOnline] (ScriptExecutionContext& context) { downcast(context).dispatchEvent(Event::create(isOnline ? eventNames().onlineEvent : eventNames().offlineEvent, false, false)); }); } void WorkerMessagingProxy::workerGlobalScopeDestroyed() { m_scriptExecutionContext->postTask([this] (ScriptExecutionContext&) { workerGlobalScopeDestroyedInternal(); }); // Will execute workerGlobalScopeDestroyedInternal() on context's thread. } void WorkerMessagingProxy::workerGlobalScopeClosed() { // Executes terminateWorkerGlobalScope() on parent context's thread. m_scriptExecutionContext->postTask([this] (ScriptExecutionContext&) { terminateWorkerGlobalScope(); }); } void WorkerMessagingProxy::workerGlobalScopeDestroyedInternal() { // WorkerGlobalScopeDestroyedTask is always the last to be performed, so the proxy is not needed for communication // in either side any more. However, the Worker object may still exist, and it assumes that the proxy exists, too. m_askedToTerminate = true; m_workerThread = nullptr; if (m_mayBeDestroyed) delete this; } void WorkerMessagingProxy::terminateWorkerGlobalScope() { if (m_askedToTerminate) return; m_askedToTerminate = true; if (m_workerThread) m_workerThread->stop(); } void WorkerMessagingProxy::confirmMessageFromWorkerObject(bool hasPendingActivity) { m_scriptExecutionContext->postTask([this, hasPendingActivity] (ScriptExecutionContext&) { reportPendingActivityInternal(true, hasPendingActivity); }); // Will execute reportPendingActivityInternal() on context's thread. } void WorkerMessagingProxy::reportPendingActivity(bool hasPendingActivity) { m_scriptExecutionContext->postTask([this, hasPendingActivity] (ScriptExecutionContext&) { reportPendingActivityInternal(false, hasPendingActivity); }); // Will execute reportPendingActivityInternal() on context's thread. } void WorkerMessagingProxy::reportPendingActivityInternal(bool confirmingMessage, bool hasPendingActivity) { if (confirmingMessage && !m_askedToTerminate) { ASSERT(m_unconfirmedMessageCount); --m_unconfirmedMessageCount; } m_workerThreadHadPendingActivity = hasPendingActivity; } bool WorkerMessagingProxy::hasPendingActivity() const { return (m_unconfirmedMessageCount || m_workerThreadHadPendingActivity) && !m_askedToTerminate; } } // namespace WebCore