blob: 0a8121ae750050b37071328129c074674a3595f0 [file] [log] [blame]
/*
* Copyright (C) 2013-2019 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:
* 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. AND ITS 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 APPLE INC. OR ITS 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.
*/
#pragma once
#import "AuxiliaryProcess.h"
#import "WebKit2Initialize.h"
#import <JavaScriptCore/ExecutableAllocator.h>
#import <wtf/OSObjectPtr.h>
#import <wtf/spi/darwin/XPCSPI.h>
// FIXME: This should be moved to an SPI header.
#if USE(APPLE_INTERNAL_SDK)
#include <os/voucher_private.h>
#else
extern "C" OS_NOTHROW void voucher_replace_default_voucher(void);
#endif
#define WEBCONTENT_SERVICE_INITIALIZER WebContentServiceInitializer
#define NETWORK_SERVICE_INITIALIZER NetworkServiceInitializer
#define PLUGIN_SERVICE_INITIALIZER PluginServiceInitializer
#define GPU_SERVICE_INITIALIZER GPUServiceInitializer
#define WEBAUTHN_SERVICE_INITIALIZER WebAuthnServiceInitializer
namespace WebKit {
class XPCServiceInitializerDelegate {
public:
XPCServiceInitializerDelegate(OSObjectPtr<xpc_connection_t> connection, xpc_object_t initializerMessage)
: m_connection(WTFMove(connection))
, m_initializerMessage(initializerMessage)
{
}
virtual ~XPCServiceInitializerDelegate();
virtual bool checkEntitlements();
virtual bool getConnectionIdentifier(IPC::Connection::Identifier& identifier);
virtual bool getProcessIdentifier(WebCore::ProcessIdentifier&);
virtual bool getClientIdentifier(String& clientIdentifier);
virtual bool getClientBundleIdentifier(String& clientBundleIdentifier);
virtual bool getClientSDKVersion(uint32_t& clientSDKVersion);
virtual bool getClientProcessName(String& clientProcessName);
virtual bool getExtraInitializationData(HashMap<String, String>& extraInitializationData);
protected:
bool hasEntitlement(const char* entitlement);
bool isClientSandboxed();
OSObjectPtr<xpc_connection_t> m_connection;
xpc_object_t m_initializerMessage;
};
template<typename XPCServiceType>
void initializeAuxiliaryProcess(AuxiliaryProcessInitializationParameters&& parameters)
{
XPCServiceType::singleton().initialize(WTFMove(parameters));
}
#if PLATFORM(MAC)
OSObjectPtr<os_transaction_t>& osTransaction();
#endif
template<typename XPCServiceType, typename XPCServiceInitializerDelegateType>
void XPCServiceInitializer(OSObjectPtr<xpc_connection_t> connection, xpc_object_t initializerMessage, xpc_object_t priorityBoostMessage)
{
if (initializerMessage) {
if (xpc_dictionary_get_bool(initializerMessage, "configure-jsc-for-testing"))
JSC::Config::configureForTesting();
if (xpc_dictionary_get_bool(initializerMessage, "disable-jit"))
JSC::ExecutableAllocator::setJITEnabled(false);
if (xpc_dictionary_get_bool(initializerMessage, "enable-shared-array-buffer")) {
JSC::Options::initialize();
JSC::Options::AllowUnfinalizedAccessScope scope;
JSC::Options::useSharedArrayBuffer() = true;
}
}
XPCServiceInitializerDelegateType delegate(WTFMove(connection), initializerMessage);
// We don't want XPC to be in charge of whether the process should be terminated or not,
// so ensure that we have an outstanding transaction here. This is not needed on iOS because
// the UIProcess takes process assertions on behalf of its child processes.
#if PLATFORM(MAC)
osTransaction() = adoptOSObject(os_transaction_create("WebKit XPC Service"));
#endif
InitializeWebKit2();
if (!delegate.checkEntitlements())
exit(EXIT_FAILURE);
AuxiliaryProcessInitializationParameters parameters;
if (priorityBoostMessage)
parameters.priorityBoostMessage = priorityBoostMessage;
if (!delegate.getConnectionIdentifier(parameters.connectionIdentifier))
exit(EXIT_FAILURE);
if (!delegate.getClientIdentifier(parameters.clientIdentifier))
exit(EXIT_FAILURE);
// The host process may not have a bundle identifier (e.g. a command line app), so don't require one.
delegate.getClientBundleIdentifier(parameters.clientBundleIdentifier);
if (!delegate.getClientSDKVersion(parameters.clientSDKVersion))
exit(EXIT_FAILURE);
WebCore::ProcessIdentifier processIdentifier;
if (!delegate.getProcessIdentifier(processIdentifier))
exit(EXIT_FAILURE);
parameters.processIdentifier = processIdentifier;
if (!delegate.getClientProcessName(parameters.uiProcessName))
exit(EXIT_FAILURE);
if (!delegate.getExtraInitializationData(parameters.extraInitializationData))
exit(EXIT_FAILURE);
// Set the task default voucher to the current value (as propagated by XPC).
voucher_replace_default_voucher();
#if HAVE(QOS_CLASSES)
if (parameters.extraInitializationData.contains("always-runs-at-background-priority"_s))
Thread::setGlobalMaxQOSClass(QOS_CLASS_UTILITY);
#endif
parameters.processType = XPCServiceType::processType;
initializeAuxiliaryProcess<XPCServiceType>(WTFMove(parameters));
}
int XPCServiceMain(int, const char**);
void XPCServiceExit(OSObjectPtr<xpc_object_t>&& priorityBoostMessage);
} // namespace WebKit