blob: 99bfb62a528db8e47ca4207e2beb8efc0fe9104d [file] [log] [blame]
/*
* Copyright (C) 2009 Google Inc. All rights reserved.
* Copyright (C) 2016 Apple 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.
*/
#include "config.h"
#include "DedicatedWorkerGlobalScope.h"
#include "ContentSecurityPolicyResponseHeaders.h"
#include "DOMWindow.h"
#include "DedicatedWorkerThread.h"
#include "JSRTCRtpScriptTransformer.h"
#include "JSRTCRtpScriptTransformerConstructor.h"
#include "MessageEvent.h"
#include "RTCRtpScriptTransformer.h"
#include "RequestAnimationFrameCallback.h"
#include "SecurityOrigin.h"
#if ENABLE(OFFSCREEN_CANVAS)
#include "WorkerAnimationController.h"
#endif
#include "WorkerObjectProxy.h"
#include <wtf/IsoMallocInlines.h>
namespace WebCore {
WTF_MAKE_ISO_ALLOCATED_IMPL(DedicatedWorkerGlobalScope);
Ref<DedicatedWorkerGlobalScope> DedicatedWorkerGlobalScope::create(const WorkerParameters& params, Ref<SecurityOrigin>&& origin, DedicatedWorkerThread& thread, Ref<SecurityOrigin>&& topOrigin, IDBClient::IDBConnectionProxy* connectionProxy, SocketProvider* socketProvider)
{
auto context = adoptRef(*new DedicatedWorkerGlobalScope(params, WTFMove(origin), thread, WTFMove(topOrigin), connectionProxy, socketProvider));
if (!params.shouldBypassMainWorldContentSecurityPolicy)
context->applyContentSecurityPolicyResponseHeaders(params.contentSecurityPolicyResponseHeaders);
return context;
}
DedicatedWorkerGlobalScope::DedicatedWorkerGlobalScope(const WorkerParameters& params, Ref<SecurityOrigin>&& origin, DedicatedWorkerThread& thread, Ref<SecurityOrigin>&& topOrigin, IDBClient::IDBConnectionProxy* connectionProxy, SocketProvider* socketProvider)
: WorkerGlobalScope(WorkerThreadType::DedicatedWorker, params, WTFMove(origin), thread, WTFMove(topOrigin), connectionProxy, socketProvider)
, m_name(params.name)
{
}
DedicatedWorkerGlobalScope::~DedicatedWorkerGlobalScope() = default;
EventTargetInterface DedicatedWorkerGlobalScope::eventTargetInterface() const
{
return DedicatedWorkerGlobalScopeEventTargetInterfaceType;
}
void DedicatedWorkerGlobalScope::prepareForDestruction()
{
WorkerGlobalScope::prepareForDestruction();
#if ENABLE(WEB_RTC)
m_rtcRtpTransformerConstructorMap.clear();
#endif
}
ExceptionOr<void> DedicatedWorkerGlobalScope::postMessage(JSC::JSGlobalObject& state, JSC::JSValue messageValue, PostMessageOptions&& options)
{
Vector<RefPtr<MessagePort>> ports;
auto message = SerializedScriptValue::create(state, messageValue, WTFMove(options.transfer), ports, SerializationContext::WorkerPostMessage);
if (message.hasException())
return message.releaseException();
// Disentangle the port in preparation for sending it to the remote context.
auto channels = MessagePort::disentanglePorts(WTFMove(ports));
if (channels.hasException())
return channels.releaseException();
thread().workerObjectProxy().postMessageToWorkerObject({ message.releaseReturnValue(), channels.releaseReturnValue() });
return { };
}
ExceptionOr<void> DedicatedWorkerGlobalScope::importScripts(const Vector<String>& urls)
{
auto result = Base::importScripts(urls);
thread().workerObjectProxy().reportPendingActivity(hasPendingActivity());
return result;
}
DedicatedWorkerThread& DedicatedWorkerGlobalScope::thread()
{
return static_cast<DedicatedWorkerThread&>(Base::thread());
}
#if ENABLE(OFFSCREEN_CANVAS)
CallbackId DedicatedWorkerGlobalScope::requestAnimationFrame(Ref<RequestAnimationFrameCallback>&& callback)
{
if (!m_workerAnimationController)
m_workerAnimationController = WorkerAnimationController::create(*this);
return m_workerAnimationController->requestAnimationFrame(WTFMove(callback));
}
void DedicatedWorkerGlobalScope::cancelAnimationFrame(CallbackId callbackId)
{
if (m_workerAnimationController)
m_workerAnimationController->cancelAnimationFrame(callbackId);
}
#endif
#if ENABLE(WEB_RTC)
ExceptionOr<void> DedicatedWorkerGlobalScope::registerRTCRtpScriptTransformer(String&& name, Ref<JSRTCRtpScriptTransformerConstructor>&& transformerConstructor)
{
ASSERT(!isMainThread());
if (name.isEmpty())
return Exception { NotSupportedError, "Name cannot be the empty string"_s };
if (m_rtcRtpTransformerConstructorMap.contains(name))
return Exception { NotSupportedError, "A transformer was already registered with this name"_s };
JSC::JSObject* jsConstructor = transformerConstructor->callbackData()->callback();
auto* globalObject = jsConstructor->globalObject();
auto& vm = globalObject->vm();
auto scope = DECLARE_THROW_SCOPE(vm);
if (!jsConstructor->isConstructor(vm))
return Exception { TypeError, "Class definitition passed to registerRTCRtpScriptTransformer() is not a constructor"_s };
auto prototype = jsConstructor->getPrototype(vm, globalObject);
RETURN_IF_EXCEPTION(scope, Exception { ExistingExceptionError });
if (!prototype.isObject())
return Exception { TypeError, "Class definitition passed to registerRTCRtpScriptTransformer() has invalid prototype"_s };
m_rtcRtpTransformerConstructorMap.add(name, WTFMove(transformerConstructor));
thread().workerObjectProxy().postTaskToWorkerObject([name = name.isolatedCopy()](auto& worker) mutable {
worker.addRTCRtpScriptTransformer(WTFMove(name));
});
return { };
}
RefPtr<RTCRtpScriptTransformer> DedicatedWorkerGlobalScope::createRTCRtpScriptTransformer(String&& name, TransferredMessagePort port)
{
auto constructor = m_rtcRtpTransformerConstructorMap.get(name);
ASSERT(constructor);
if (!constructor)
return nullptr;
auto* jsConstructor = constructor->callbackData()->callback();
auto* globalObject = constructor->callbackData()->globalObject();
auto& vm = globalObject->vm();
auto scope = DECLARE_CATCH_SCOPE(vm);
JSC::JSLockHolder lock { globalObject };
m_pendingRTCTransfomerMessagePort = MessagePort::entangle(*this, WTFMove(port));
JSC::MarkedArgumentBuffer args;
auto* object = JSC::construct(globalObject, jsConstructor, args, "Failed to construct RTCRtpScriptTransformer");
ASSERT(!!scope.exception() == !object);
if (scope.exception()) {
scope.clearException();
return nullptr;
}
RETURN_IF_EXCEPTION(scope, nullptr);
auto& jsTransformer = *JSC::jsCast<JSRTCRtpScriptTransformer*>(object);
auto& transformer = jsTransformer.wrapped();
transformer.setCallback(makeUnique<JSCallbackDataStrong>(&jsTransformer, globalObject));
return &transformer;
}
#endif
} // namespace WebCore