ProcessLauncherGtk.cpp [plain text]
#include "config.h"
#include "ProcessLauncher.h"
#include "Connection.h"
#include "ProcessExecutablePath.h"
#include <WebCore/AuthenticationChallenge.h>
#include <WebCore/FileSystem.h>
#include <WebCore/NetworkingContext.h>
#include <WebCore/ResourceHandle.h>
#include <errno.h>
#include <fcntl.h>
#include <glib.h>
#include <locale.h>
#include <wtf/RunLoop.h>
#include <wtf/text/CString.h>
#include <wtf/text/WTFString.h>
#include <wtf/gobject/GUniquePtr.h>
#include <wtf/gobject/GlibUtilities.h>
using namespace WebCore;
namespace WebKit {
static void childSetupFunction(gpointer userData)
{
int socket = GPOINTER_TO_INT(userData);
close(socket);
}
void ProcessLauncher::launchProcess()
{
GPid pid = 0;
IPC::Connection::SocketPair socketPair = IPC::Connection::createPlatformConnection(IPC::Connection::ConnectionOptions::SetCloexecOnServer);
String executablePath, pluginPath;
CString realExecutablePath, realPluginPath;
switch (m_launchOptions.processType) {
case WebProcess:
executablePath = executablePathOfWebProcess();
break;
case PluginProcess:
executablePath = executablePathOfPluginProcess();
if (m_launchOptions.extraInitializationData.contains("requires-gtk2"))
executablePath.append('2');
pluginPath = m_launchOptions.extraInitializationData.get("plugin-path");
realPluginPath = fileSystemRepresentation(pluginPath);
break;
#if ENABLE(NETWORK_PROCESS)
case NetworkProcess:
executablePath = executablePathOfNetworkProcess();
break;
#endif
default:
ASSERT_NOT_REACHED();
return;
}
realExecutablePath = fileSystemRepresentation(executablePath);
GUniquePtr<gchar> socket(g_strdup_printf("%d", socketPair.client));
unsigned nargs = 4;
#ifndef NDEBUG
Vector<CString> prefixArgs;
if (!m_launchOptions.processCmdPrefix.isNull()) {
Vector<String> splitArgs;
m_launchOptions.processCmdPrefix.split(' ', splitArgs);
for (auto it = splitArgs.begin(); it != splitArgs.end(); it++)
prefixArgs.append(it->utf8());
nargs += prefixArgs.size();
}
#endif
char** argv = g_newa(char*, nargs);
unsigned i = 0;
#ifndef NDEBUG
for (auto it = prefixArgs.begin(); it != prefixArgs.end(); it++)
argv[i++] = const_cast<char*>(it->data());
#endif
argv[i++] = const_cast<char*>(realExecutablePath.data());
argv[i++] = socket.get();
argv[i++] = const_cast<char*>(realPluginPath.data());
argv[i++] = 0;
GUniqueOutPtr<GError> error;
if (!g_spawn_async(0, argv, 0, G_SPAWN_LEAVE_DESCRIPTORS_OPEN, childSetupFunction, GINT_TO_POINTER(socketPair.server), &pid, &error.outPtr())) {
g_printerr("Unable to fork a new WebProcess: %s.\n", error->message);
ASSERT_NOT_REACHED();
}
while (fcntl(socketPair.client, F_SETFD, FD_CLOEXEC) == -1)
RELEASE_ASSERT(errno != EINTR);
close(socketPair.client);
m_processIdentifier = pid;
RunLoop::main().dispatch(bind(&ProcessLauncher::didFinishLaunchingProcess, this, m_processIdentifier, socketPair.server));
}
void ProcessLauncher::terminateProcess()
{
if (m_isLaunching) {
invalidate();
return;
}
if (!m_processIdentifier)
return;
kill(m_processIdentifier, SIGKILL);
m_processIdentifier = 0;
}
void ProcessLauncher::platformInvalidate()
{
}
}