| 1 | /* | 
|---|
| 2 | * Copyright (C) 2008-2017 Apple Inc. All Rights Reserved. | 
|---|
| 3 | * Copyright (C) 2009 Google Inc. All Rights Reserved. | 
|---|
| 4 | * | 
|---|
| 5 | * Redistribution and use in source and binary forms, with or without | 
|---|
| 6 | * modification, are permitted provided that the following conditions | 
|---|
| 7 | * are met: | 
|---|
| 8 | * 1. Redistributions of source code must retain the above copyright | 
|---|
| 9 | *    notice, this list of conditions and the following disclaimer. | 
|---|
| 10 | * 2. Redistributions in binary form must reproduce the above copyright | 
|---|
| 11 | *    notice, this list of conditions and the following disclaimer in the | 
|---|
| 12 | *    documentation and/or other materials provided with the distribution. | 
|---|
| 13 | * | 
|---|
| 14 | * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY | 
|---|
| 15 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | 
|---|
| 16 | * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR | 
|---|
| 17 | * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE INC. OR | 
|---|
| 18 | * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, | 
|---|
| 19 | * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, | 
|---|
| 20 | * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR | 
|---|
| 21 | * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY | 
|---|
| 22 | * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | 
|---|
| 23 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | 
|---|
| 24 | * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 
|---|
| 25 | * | 
|---|
| 26 | */ | 
|---|
| 27 |  | 
|---|
| 28 | #include "config.h" | 
|---|
| 29 | #include "WorkerMessagingProxy.h" | 
|---|
| 30 |  | 
|---|
| 31 | #include "CacheStorageProvider.h" | 
|---|
| 32 | #include "ContentSecurityPolicy.h" | 
|---|
| 33 | #include "DOMWindow.h" | 
|---|
| 34 | #include "DedicatedWorkerGlobalScope.h" | 
|---|
| 35 | #include "DedicatedWorkerThread.h" | 
|---|
| 36 | #include "Document.h" | 
|---|
| 37 | #include "ErrorEvent.h" | 
|---|
| 38 | #include "EventNames.h" | 
|---|
| 39 | #include "MessageEvent.h" | 
|---|
| 40 | #include "Page.h" | 
|---|
| 41 | #include "ScriptExecutionContext.h" | 
|---|
| 42 | #include "Worker.h" | 
|---|
| 43 | #include "WorkerInspectorProxy.h" | 
|---|
| 44 | #include <JavaScriptCore/ConsoleTypes.h> | 
|---|
| 45 | #include <JavaScriptCore/ScriptCallStack.h> | 
|---|
| 46 | #include <wtf/MainThread.h> | 
|---|
| 47 | #include <wtf/RunLoop.h> | 
|---|
| 48 |  | 
|---|
| 49 | namespace WebCore { | 
|---|
| 50 |  | 
|---|
| 51 | WorkerGlobalScopeProxy& WorkerGlobalScopeProxy::create(Worker& worker) | 
|---|
| 52 | { | 
|---|
| 53 | return *new WorkerMessagingProxy(worker); | 
|---|
| 54 | } | 
|---|
| 55 |  | 
|---|
| 56 | WorkerMessagingProxy::WorkerMessagingProxy(Worker& workerObject) | 
|---|
| 57 | : m_scriptExecutionContext(workerObject.scriptExecutionContext()) | 
|---|
| 58 | , m_inspectorProxy(std::make_unique<WorkerInspectorProxy>(workerObject.identifier())) | 
|---|
| 59 | , m_workerObject(&workerObject) | 
|---|
| 60 | { | 
|---|
| 61 | ASSERT((is<Document>(*m_scriptExecutionContext) && isMainThread()) | 
|---|
| 62 | || (is<WorkerGlobalScope>(*m_scriptExecutionContext) && downcast<WorkerGlobalScope>(*m_scriptExecutionContext).thread().thread() == &Thread::current())); | 
|---|
| 63 |  | 
|---|
| 64 | // Nobody outside this class ref counts this object. The original ref | 
|---|
| 65 | // is balanced by the deref in workerGlobalScopeDestroyedInternal. | 
|---|
| 66 | } | 
|---|
| 67 |  | 
|---|
| 68 | WorkerMessagingProxy::~WorkerMessagingProxy() | 
|---|
| 69 | { | 
|---|
| 70 | ASSERT(!m_workerObject); | 
|---|
| 71 | ASSERT((is<Document>(*m_scriptExecutionContext) && isMainThread()) | 
|---|
| 72 | || (is<WorkerGlobalScope>(*m_scriptExecutionContext) && downcast<WorkerGlobalScope>(*m_scriptExecutionContext).thread().thread() == &Thread::current())); | 
|---|
| 73 | } | 
|---|
| 74 |  | 
|---|
| 75 | void WorkerMessagingProxy::(const URL& scriptURL, const String& name, const String& userAgent, bool isOnline, const String& sourceCode, const ContentSecurityPolicyResponseHeaders& , bool shouldBypassMainWorldContentSecurityPolicy, MonotonicTime timeOrigin, JSC::RuntimeFlags runtimeFlags, PAL::SessionID sessionID) | 
|---|
| 76 | { | 
|---|
| 77 | // FIXME: This need to be revisited when we support nested worker one day | 
|---|
| 78 | ASSERT(m_scriptExecutionContext); | 
|---|
| 79 | Document& document = downcast<Document>(*m_scriptExecutionContext); | 
|---|
| 80 | WorkerThreadStartMode startMode = m_inspectorProxy->workerStartMode(*m_scriptExecutionContext.get()); | 
|---|
| 81 | String identifier = m_inspectorProxy->identifier(); | 
|---|
| 82 |  | 
|---|
| 83 | #if ENABLE(INDEXED_DATABASE) | 
|---|
| 84 | IDBClient::IDBConnectionProxy* proxy = document.idbConnectionProxy(); | 
|---|
| 85 | #else | 
|---|
| 86 | IDBClient::IDBConnectionProxy* proxy = nullptr; | 
|---|
| 87 | #endif | 
|---|
| 88 |  | 
|---|
| 89 | SocketProvider* socketProvider = document.socketProvider(); | 
|---|
| 90 |  | 
|---|
| 91 | auto thread = DedicatedWorkerThread::create(scriptURL, name, identifier, userAgent, isOnline, sourceCode, *this, *this, *this, startMode, contentSecurityPolicyResponseHeaders, shouldBypassMainWorldContentSecurityPolicy, document.topOrigin(), timeOrigin, proxy, socketProvider, runtimeFlags, sessionID); | 
|---|
| 92 |  | 
|---|
| 93 | workerThreadCreated(thread.get()); | 
|---|
| 94 | thread->start(nullptr); | 
|---|
| 95 |  | 
|---|
| 96 | m_inspectorProxy->workerStarted(m_scriptExecutionContext.get(), thread.ptr(), scriptURL); | 
|---|
| 97 | } | 
|---|
| 98 |  | 
|---|
| 99 | void WorkerMessagingProxy::postMessageToWorkerObject(MessageWithMessagePorts&& message) | 
|---|
| 100 | { | 
|---|
| 101 | m_scriptExecutionContext->postTask([this, message = WTFMove(message)] (ScriptExecutionContext& context) mutable { | 
|---|
| 102 | Worker* workerObject = this->workerObject(); | 
|---|
| 103 | if (!workerObject || askedToTerminate()) | 
|---|
| 104 | return; | 
|---|
| 105 |  | 
|---|
| 106 | auto ports = MessagePort::entanglePorts(context, WTFMove(message.transferredPorts)); | 
|---|
| 107 | workerObject->dispatchEvent(MessageEvent::create(WTFMove(ports), message.message.releaseNonNull())); | 
|---|
| 108 | }); | 
|---|
| 109 | } | 
|---|
| 110 |  | 
|---|
| 111 | void WorkerMessagingProxy::postMessageToWorkerGlobalScope(MessageWithMessagePorts&& message) | 
|---|
| 112 | { | 
|---|
| 113 | if (m_askedToTerminate) | 
|---|
| 114 | return; | 
|---|
| 115 |  | 
|---|
| 116 | ScriptExecutionContext::Task task([message = WTFMove(message)] (ScriptExecutionContext& scriptContext) mutable { | 
|---|
| 117 | ASSERT_WITH_SECURITY_IMPLICATION(scriptContext.isWorkerGlobalScope()); | 
|---|
| 118 | auto& context = static_cast<DedicatedWorkerGlobalScope&>(scriptContext); | 
|---|
| 119 | auto ports = MessagePort::entanglePorts(scriptContext, WTFMove(message.transferredPorts)); | 
|---|
| 120 | context.dispatchEvent(MessageEvent::create(WTFMove(ports), message.message.releaseNonNull())); | 
|---|
| 121 | context.thread().workerObjectProxy().confirmMessageFromWorkerObject(context.hasPendingActivity()); | 
|---|
| 122 | }); | 
|---|
| 123 |  | 
|---|
| 124 | if (m_workerThread) { | 
|---|
| 125 | ++m_unconfirmedMessageCount; | 
|---|
| 126 | m_workerThread->runLoop().postTask(WTFMove(task)); | 
|---|
| 127 | } else | 
|---|
| 128 | m_queuedEarlyTasks.append(std::make_unique<ScriptExecutionContext::Task>(WTFMove(task))); | 
|---|
| 129 | } | 
|---|
| 130 |  | 
|---|
| 131 | void WorkerMessagingProxy::postTaskToLoader(ScriptExecutionContext::Task&& task) | 
|---|
| 132 | { | 
|---|
| 133 | // FIXME: In case of nested workers, this should go directly to the root Document context. | 
|---|
| 134 | ASSERT(m_scriptExecutionContext->isDocument()); | 
|---|
| 135 | m_scriptExecutionContext->postTask(WTFMove(task)); | 
|---|
| 136 | } | 
|---|
| 137 |  | 
|---|
| 138 | Ref<CacheStorageConnection> WorkerMessagingProxy::createCacheStorageConnection() | 
|---|
| 139 | { | 
|---|
| 140 | ASSERT(isMainThread()); | 
|---|
| 141 | auto& document = downcast<Document>(*m_scriptExecutionContext); | 
|---|
| 142 | return document.page()->cacheStorageProvider().createCacheStorageConnection(document.page()->sessionID()); | 
|---|
| 143 | } | 
|---|
| 144 |  | 
|---|
| 145 | bool WorkerMessagingProxy::postTaskForModeToWorkerGlobalScope(ScriptExecutionContext::Task&& task, const String& mode) | 
|---|
| 146 | { | 
|---|
| 147 | if (m_askedToTerminate) | 
|---|
| 148 | return false; | 
|---|
| 149 |  | 
|---|
| 150 | ASSERT(m_workerThread); | 
|---|
| 151 | m_workerThread->runLoop().postTaskForMode(WTFMove(task), mode); | 
|---|
| 152 | return true; | 
|---|
| 153 | } | 
|---|
| 154 |  | 
|---|
| 155 | void WorkerMessagingProxy::postExceptionToWorkerObject(const String& errorMessage, int lineNumber, int columnNumber, const String& sourceURL) | 
|---|
| 156 | { | 
|---|
| 157 | m_scriptExecutionContext->postTask([this, errorMessage = errorMessage.isolatedCopy(), sourceURL = sourceURL.isolatedCopy(), lineNumber, columnNumber] (ScriptExecutionContext& context) { | 
|---|
| 158 | Worker* workerObject = this->workerObject(); | 
|---|
| 159 | if (!workerObject) | 
|---|
| 160 | return; | 
|---|
| 161 |  | 
|---|
| 162 | // We don't bother checking the askedToTerminate() flag here, because exceptions should *always* be reported even if the thread is terminated. | 
|---|
| 163 | // 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. | 
|---|
| 164 |  | 
|---|
| 165 | auto event = ErrorEvent::create(errorMessage, sourceURL, lineNumber, columnNumber, { }); | 
|---|
| 166 | workerObject->dispatchEvent(event); | 
|---|
| 167 | if (!event->defaultPrevented()) | 
|---|
| 168 | context.reportException(errorMessage, lineNumber, columnNumber, sourceURL, nullptr, nullptr); | 
|---|
| 169 | }); | 
|---|
| 170 | } | 
|---|
| 171 |  | 
|---|
| 172 | void WorkerMessagingProxy::postMessageToDebugger(const String& message) | 
|---|
| 173 | { | 
|---|
| 174 | RunLoop::main().dispatch([this, protectedThis = makeRef(*this), message = message.isolatedCopy()] { | 
|---|
| 175 | if (!m_mayBeDestroyed) | 
|---|
| 176 | m_inspectorProxy->sendMessageFromWorkerToFrontend(message); | 
|---|
| 177 | }); | 
|---|
| 178 | } | 
|---|
| 179 |  | 
|---|
| 180 | void WorkerMessagingProxy::setResourceCachingDisabled(bool disabled) | 
|---|
| 181 | { | 
|---|
| 182 | postTaskToLoader([disabled] (ScriptExecutionContext& context) { | 
|---|
| 183 | ASSERT(isMainThread()); | 
|---|
| 184 | if (auto* page = downcast<Document>(context).page()) | 
|---|
| 185 | page->setResourceCachingDisabled(disabled); | 
|---|
| 186 | }); | 
|---|
| 187 | } | 
|---|
| 188 |  | 
|---|
| 189 | void WorkerMessagingProxy::workerThreadCreated(DedicatedWorkerThread& workerThread) | 
|---|
| 190 | { | 
|---|
| 191 | m_workerThread = &workerThread; | 
|---|
| 192 |  | 
|---|
| 193 | if (m_askedToTerminate) { | 
|---|
| 194 | // Worker.terminate() could be called from JS before the thread was created. | 
|---|
| 195 | m_workerThread->stop(nullptr); | 
|---|
| 196 | } else { | 
|---|
| 197 | ASSERT(!m_unconfirmedMessageCount); | 
|---|
| 198 | m_unconfirmedMessageCount = m_queuedEarlyTasks.size(); | 
|---|
| 199 | m_workerThreadHadPendingActivity = true; // Worker initialization means a pending activity. | 
|---|
| 200 |  | 
|---|
| 201 | auto queuedEarlyTasks = WTFMove(m_queuedEarlyTasks); | 
|---|
| 202 | for (auto& task : queuedEarlyTasks) | 
|---|
| 203 | m_workerThread->runLoop().postTask(WTFMove(*task)); | 
|---|
| 204 | } | 
|---|
| 205 | } | 
|---|
| 206 |  | 
|---|
| 207 | void WorkerMessagingProxy::workerObjectDestroyed() | 
|---|
| 208 | { | 
|---|
| 209 | m_workerObject = nullptr; | 
|---|
| 210 | m_scriptExecutionContext->postTask([this] (ScriptExecutionContext&) { | 
|---|
| 211 | m_mayBeDestroyed = true; | 
|---|
| 212 | if (m_workerThread) | 
|---|
| 213 | terminateWorkerGlobalScope(); | 
|---|
| 214 | else | 
|---|
| 215 | workerGlobalScopeDestroyedInternal(); | 
|---|
| 216 | }); | 
|---|
| 217 | } | 
|---|
| 218 |  | 
|---|
| 219 | void WorkerMessagingProxy::notifyNetworkStateChange(bool isOnline) | 
|---|
| 220 | { | 
|---|
| 221 | if (m_askedToTerminate) | 
|---|
| 222 | return; | 
|---|
| 223 |  | 
|---|
| 224 | if (!m_workerThread) | 
|---|
| 225 | return; | 
|---|
| 226 |  | 
|---|
| 227 | m_workerThread->runLoop().postTask([isOnline] (ScriptExecutionContext& context) { | 
|---|
| 228 | auto& globalScope = downcast<WorkerGlobalScope>(context); | 
|---|
| 229 | globalScope.setIsOnline(isOnline); | 
|---|
| 230 | globalScope.dispatchEvent(Event::create(isOnline ? eventNames().onlineEvent : eventNames().offlineEvent, Event::CanBubble::No, Event::IsCancelable::No)); | 
|---|
| 231 | }); | 
|---|
| 232 | } | 
|---|
| 233 |  | 
|---|
| 234 | void WorkerMessagingProxy::workerGlobalScopeDestroyed() | 
|---|
| 235 | { | 
|---|
| 236 | m_scriptExecutionContext->postTask([this] (ScriptExecutionContext&) { | 
|---|
| 237 | workerGlobalScopeDestroyedInternal(); | 
|---|
| 238 | }); | 
|---|
| 239 | } | 
|---|
| 240 |  | 
|---|
| 241 | void WorkerMessagingProxy::workerGlobalScopeClosed() | 
|---|
| 242 | { | 
|---|
| 243 | m_scriptExecutionContext->postTask([this] (ScriptExecutionContext&) { | 
|---|
| 244 | terminateWorkerGlobalScope(); | 
|---|
| 245 | }); | 
|---|
| 246 | } | 
|---|
| 247 |  | 
|---|
| 248 | void WorkerMessagingProxy::workerGlobalScopeDestroyedInternal() | 
|---|
| 249 | { | 
|---|
| 250 | // This is always the last task to be performed, so the proxy is not needed for communication | 
|---|
| 251 | // in either side any more. However, the Worker object may still exist, and it assumes that the proxy exists, too. | 
|---|
| 252 | m_askedToTerminate = true; | 
|---|
| 253 | m_workerThread = nullptr; | 
|---|
| 254 |  | 
|---|
| 255 | m_inspectorProxy->workerTerminated(); | 
|---|
| 256 |  | 
|---|
| 257 | // This balances the original ref in construction. | 
|---|
| 258 | if (m_mayBeDestroyed) | 
|---|
| 259 | deref(); | 
|---|
| 260 | } | 
|---|
| 261 |  | 
|---|
| 262 | void WorkerMessagingProxy::terminateWorkerGlobalScope() | 
|---|
| 263 | { | 
|---|
| 264 | if (m_askedToTerminate) | 
|---|
| 265 | return; | 
|---|
| 266 | m_askedToTerminate = true; | 
|---|
| 267 |  | 
|---|
| 268 | m_inspectorProxy->workerTerminated(); | 
|---|
| 269 |  | 
|---|
| 270 | if (m_workerThread) | 
|---|
| 271 | m_workerThread->stop(nullptr); | 
|---|
| 272 | } | 
|---|
| 273 |  | 
|---|
| 274 | void WorkerMessagingProxy::confirmMessageFromWorkerObject(bool hasPendingActivity) | 
|---|
| 275 | { | 
|---|
| 276 | m_scriptExecutionContext->postTask([this, hasPendingActivity] (ScriptExecutionContext&) { | 
|---|
| 277 | reportPendingActivityInternal(true, hasPendingActivity); | 
|---|
| 278 | }); | 
|---|
| 279 | } | 
|---|
| 280 |  | 
|---|
| 281 | void WorkerMessagingProxy::reportPendingActivity(bool hasPendingActivity) | 
|---|
| 282 | { | 
|---|
| 283 | m_scriptExecutionContext->postTask([this, hasPendingActivity] (ScriptExecutionContext&) { | 
|---|
| 284 | reportPendingActivityInternal(false, hasPendingActivity); | 
|---|
| 285 | }); | 
|---|
| 286 | } | 
|---|
| 287 |  | 
|---|
| 288 | void WorkerMessagingProxy::reportPendingActivityInternal(bool confirmingMessage, bool hasPendingActivity) | 
|---|
| 289 | { | 
|---|
| 290 | if (confirmingMessage && !m_askedToTerminate) { | 
|---|
| 291 | ASSERT(m_unconfirmedMessageCount); | 
|---|
| 292 | --m_unconfirmedMessageCount; | 
|---|
| 293 | } | 
|---|
| 294 |  | 
|---|
| 295 | m_workerThreadHadPendingActivity = hasPendingActivity; | 
|---|
| 296 | } | 
|---|
| 297 |  | 
|---|
| 298 | bool WorkerMessagingProxy::hasPendingActivity() const | 
|---|
| 299 | { | 
|---|
| 300 | return (m_unconfirmedMessageCount || m_workerThreadHadPendingActivity) && !m_askedToTerminate; | 
|---|
| 301 | } | 
|---|
| 302 |  | 
|---|
| 303 | } // namespace WebCore | 
|---|
| 304 |  | 
|---|