blob: 998b71eba0863cd6921ac37b90229574422e0c92 [file] [log] [blame]
/*
* Copyright (C) 2013 University of Szeged
* Copyright (C) 2013 Renata Hodovan <reni@inf.u-szeged.hu>
* Copyright (C) 2010 Apple Inc. All rights reserved.
* Copyright (C) 2010 Nokia Corporation and/or its subsidiary(-ies)
*
* 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.
*/
#include "config.h"
#include "WebProcess.h"
#include <QGuiApplication>
#include <QList>
#include <QNetworkProxyFactory>
#include <QString>
#include <QStringList>
#include <QUrl>
#include <WebCore/RunLoop.h>
#include <errno.h>
#include <runtime/InitializeThreading.h>
#include <wtf/MainThread.h>
#if USE(QTKIT)
#include "WebSystemInterface.h"
#endif
#ifndef NDEBUG
#if !OS(WINDOWS)
#include <unistd.h>
#endif
#endif
#ifndef NDEBUG
#include <QDebug>
#endif
#if OS(DARWIN) && !USE(UNIX_DOMAIN_SOCKETS)
#include <servers/bootstrap.h>
extern "C" kern_return_t bootstrap_look_up2(mach_port_t, const name_t, mach_port_t*, pid_t, uint64_t);
#endif
#if ENABLE(SUID_SANDBOX_LINUX)
#include "SandboxEnvironmentLinux.h"
#include <sys/wait.h>
#endif
using namespace WebCore;
namespace WebKit {
#ifndef NDEBUG
#if OS(WINDOWS)
static void sleep(unsigned seconds)
{
::Sleep(seconds * 1000);
}
#endif
#endif
class EnvHttpProxyFactory : public QNetworkProxyFactory
{
public:
EnvHttpProxyFactory() { }
bool initializeFromEnvironment();
QList<QNetworkProxy> queryProxy(const QNetworkProxyQuery& query = QNetworkProxyQuery());
private:
QList<QNetworkProxy> m_httpProxy;
QList<QNetworkProxy> m_httpsProxy;
};
bool EnvHttpProxyFactory::initializeFromEnvironment()
{
bool wasSetByEnvironment = false;
QUrl proxyUrl = QUrl::fromUserInput(QString::fromLocal8Bit(qgetenv("http_proxy")));
if (proxyUrl.isValid() && !proxyUrl.host().isEmpty()) {
int proxyPort = (proxyUrl.port() > 0) ? proxyUrl.port() : 8080;
m_httpProxy << QNetworkProxy(QNetworkProxy::HttpProxy, proxyUrl.host(), proxyPort);
wasSetByEnvironment = true;
} else
m_httpProxy << QNetworkProxy::NoProxy;
proxyUrl = QUrl::fromUserInput(QString::fromLocal8Bit(qgetenv("https_proxy")));
if (proxyUrl.isValid() && !proxyUrl.host().isEmpty()) {
int proxyPort = (proxyUrl.port() > 0) ? proxyUrl.port() : 8080;
m_httpsProxy << QNetworkProxy(QNetworkProxy::HttpProxy, proxyUrl.host(), proxyPort);
wasSetByEnvironment = true;
} else
m_httpsProxy << QNetworkProxy::NoProxy;
return wasSetByEnvironment;
}
QList<QNetworkProxy> EnvHttpProxyFactory::queryProxy(const QNetworkProxyQuery& query)
{
QString protocol = query.protocolTag().toLower();
bool localHost = false;
if (!query.peerHostName().compare(QLatin1String("localhost"), Qt::CaseInsensitive) || !query.peerHostName().compare(QLatin1String("127.0.0.1"), Qt::CaseInsensitive))
localHost = true;
if (protocol == QLatin1String("http") && !localHost)
return m_httpProxy;
if (protocol == QLatin1String("https") && !localHost)
return m_httpsProxy;
QList<QNetworkProxy> proxies;
proxies << QNetworkProxy::NoProxy;
return proxies;
}
static void initializeProxy()
{
QList<QNetworkProxy> proxylist = QNetworkProxyFactory::systemProxyForQuery();
if (proxylist.count() == 1) {
QNetworkProxy proxy = proxylist.first();
if (proxy == QNetworkProxy::NoProxy || proxy == QNetworkProxy::DefaultProxy) {
EnvHttpProxyFactory* proxyFactory = new EnvHttpProxyFactory();
if (proxyFactory->initializeFromEnvironment()) {
QNetworkProxyFactory::setApplicationProxyFactory(proxyFactory);
return;
}
}
}
QNetworkProxyFactory::setUseSystemConfiguration(true);
}
#if ENABLE(SUID_SANDBOX_LINUX)
pid_t chrootMe()
{
// Get the file descriptor of the socketpair.
char* sandboxSocketDescriptorString = getenv(SANDBOX_DESCRIPTOR);
if (!sandboxSocketDescriptorString)
return -1;
char* firstInvalidCharacter;
long int sandboxSocketDescriptor = strtol(sandboxSocketDescriptorString, &firstInvalidCharacter, 10);
if (*firstInvalidCharacter != '\0') {
fprintf(stderr, "The socket descriptor of sandbox is not valid.\n");
return -1;
}
// Get the PID of the setuid helper.
char* sandboxHelperPIDString = getenv(SANDBOX_HELPER_PID);
pid_t sandboxHelperPID = -1;
// If no PID is available, the default of -1 will do.
if (sandboxHelperPIDString) {
errno = 0;
sandboxHelperPID = strtol(sandboxHelperPIDString, &firstInvalidCharacter, 10);
if (*firstInvalidCharacter != '\0') {
fprintf(stderr, "The PID of sandbox is not valid.\n");
return -1;
}
}
// Send the chrootMe message to the helper.
char sandboxMeMessage = MSG_CHROOTME;
ssize_t numberOfCharacters = write(sandboxSocketDescriptor, &sandboxMeMessage, 1);
if (numberOfCharacters != 1) {
fprintf(stderr, "ChrootMe msg failed to write: %s.\n", strerror(errno));
return -1;
}
// Read the acknowledgement message from the helper.
numberOfCharacters = read(sandboxSocketDescriptor, &sandboxMeMessage, 1);
if (numberOfCharacters != 1 || sandboxMeMessage != MSG_CHROOTED) {
fprintf(stderr, "Couldn't read the confirmation message: %s.\n", strerror(errno));
return -1;
}
close(sandboxSocketDescriptor);
// Wait for the helper process.
int expectedPID = waitpid(sandboxHelperPID, 0, 0);
if (expectedPID != -1 && (sandboxHelperPID == -1 || expectedPID == sandboxHelperPID))
return expectedPID;
fprintf(stderr, "Couldn't wait for the helper process: %s\n", strerror(errno));
return -1;
}
#endif
Q_DECL_EXPORT int WebProcessMainQt(QGuiApplication* app)
{
#if ENABLE(SUID_SANDBOX_LINUX)
pid_t helper = chrootMe();
if (helper == -1) {
fprintf(stderr, "Asking for chroot failed.\n");
return -1;
}
#endif
initializeProxy();
JSC::initializeThreading();
WTF::initializeMainThread();
RunLoop::initializeMainRunLoop();
#if USE(QTKIT)
InitWebCoreSystemInterfaceForWK2();
#endif
// Create the connection.
if (app->arguments().size() <= 1) {
qDebug() << "Error: wrong number of arguments.";
return 1;
}
#if OS(DARWIN)
QString serviceName = app->arguments().value(1);
// Get the server port.
mach_port_t identifier;
kern_return_t kr = bootstrap_look_up2(bootstrap_port, serviceName.toUtf8().data(), &identifier, 0, 0);
if (kr) {
printf("bootstrap_look_up2 result: %x", kr);
return 2;
}
#else
bool wasNumber = false;
qulonglong id = app->arguments().at(1).toULongLong(&wasNumber, 10);
if (!wasNumber) {
qDebug() << "Error: connection identifier wrong.";
return 1;
}
CoreIPC::Connection::Identifier identifier;
#if OS(WINDOWS)
// Convert to HANDLE
identifier = reinterpret_cast<CoreIPC::Connection::Identifier>(id);
#else
// Convert to int
identifier = static_cast<CoreIPC::Connection::Identifier>(id);
#endif
#endif
WebKit::ChildProcessInitializationParameters parameters;
parameters.connectionIdentifier = identifier;
WebKit::WebProcess::shared().initialize(parameters);
RunLoop::run();
// FIXME: Do more cleanup here.
delete app;
return 0;
}
}