2021-07-07 11:36:03 +02:00
|
|
|
/****************************************************************************
|
|
|
|
|
**
|
|
|
|
|
** Copyright (C) 2021 The Qt Company Ltd.
|
|
|
|
|
** Contact: https://www.qt.io/licensing/
|
|
|
|
|
**
|
|
|
|
|
** This file is part of Qt Creator.
|
|
|
|
|
**
|
|
|
|
|
** Commercial License Usage
|
|
|
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
|
|
|
** accordance with the commercial license agreement provided with the
|
|
|
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
|
|
|
** a written agreement between you and The Qt Company. For licensing terms
|
|
|
|
|
** and conditions see https://www.qt.io/terms-conditions. For further
|
|
|
|
|
** information use the contact form at https://www.qt.io/contact-us.
|
|
|
|
|
**
|
|
|
|
|
** GNU General Public License Usage
|
|
|
|
|
** Alternatively, this file may be used under the terms of the GNU
|
|
|
|
|
** General Public License version 3 as published by the Free Software
|
|
|
|
|
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
|
|
|
|
|
** included in the packaging of this file. Please review the following
|
|
|
|
|
** information to ensure the GNU General Public License requirements will
|
|
|
|
|
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
|
|
|
|
|
**
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
|
|
#include "launcherinterface.h"
|
|
|
|
|
|
|
|
|
|
#include "launcherpackets.h"
|
|
|
|
|
#include "launchersocket.h"
|
|
|
|
|
|
|
|
|
|
#include <QtCore/qcoreapplication.h>
|
|
|
|
|
#include <QtCore/qdebug.h>
|
|
|
|
|
#include <QtCore/qdir.h>
|
|
|
|
|
#include <QtCore/qprocess.h>
|
|
|
|
|
#include <QtNetwork/qlocalserver.h>
|
|
|
|
|
|
|
|
|
|
#ifdef Q_OS_UNIX
|
|
|
|
|
#include <unistd.h>
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
namespace Utils {
|
|
|
|
|
|
|
|
|
|
namespace Internal {
|
|
|
|
|
|
|
|
|
|
class LauncherProcess : public QProcess
|
|
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
LauncherProcess(QObject *parent) : QProcess(parent)
|
|
|
|
|
{
|
|
|
|
|
#if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0) && defined(Q_OS_UNIX)
|
|
|
|
|
setChildProcessModifier([this] { setupChildProcess_impl(); });
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
#if QT_VERSION < QT_VERSION_CHECK(6, 0, 0)
|
|
|
|
|
void setupChildProcess() override
|
|
|
|
|
{
|
|
|
|
|
setupChildProcess_impl();
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
void setupChildProcess_impl()
|
|
|
|
|
{
|
|
|
|
|
#ifdef Q_OS_UNIX
|
|
|
|
|
const auto pid = static_cast<pid_t>(processId());
|
|
|
|
|
setpgid(pid, pid);
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static QString launcherSocketName()
|
|
|
|
|
{
|
|
|
|
|
return QStringLiteral("qtcreator_processlauncher-%1")
|
|
|
|
|
.arg(QString::number(qApp->applicationPid()));
|
|
|
|
|
}
|
|
|
|
|
|
2021-07-14 16:02:25 +02:00
|
|
|
class LauncherInterfacePrivate : public QObject
|
2021-07-07 11:36:03 +02:00
|
|
|
{
|
2021-07-14 16:02:25 +02:00
|
|
|
Q_OBJECT
|
|
|
|
|
public:
|
|
|
|
|
LauncherInterfacePrivate();
|
|
|
|
|
~LauncherInterfacePrivate() override;
|
2021-07-07 11:36:03 +02:00
|
|
|
|
2021-07-14 16:02:25 +02:00
|
|
|
void doStart();
|
|
|
|
|
void doStop();
|
|
|
|
|
void handleNewConnection();
|
|
|
|
|
void handleProcessError();
|
|
|
|
|
void handleProcessFinished();
|
|
|
|
|
void handleProcessStderr();
|
|
|
|
|
Internal::LauncherSocket *socket() const { return m_socket; }
|
|
|
|
|
|
|
|
|
|
signals:
|
|
|
|
|
void errorOccurred(const QString &error);
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
QLocalServer * const m_server;
|
|
|
|
|
Internal::LauncherSocket *const m_socket;
|
|
|
|
|
Internal::LauncherProcess *m_process = nullptr;
|
|
|
|
|
int m_startRequests = 0;
|
|
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
LauncherInterfacePrivate::LauncherInterfacePrivate()
|
|
|
|
|
: m_server(new QLocalServer(this)), m_socket(new LauncherSocket(this))
|
2021-07-07 11:36:03 +02:00
|
|
|
{
|
2021-07-14 16:02:25 +02:00
|
|
|
QObject::connect(m_server, &QLocalServer::newConnection,
|
|
|
|
|
this, &LauncherInterfacePrivate::handleNewConnection);
|
2021-07-07 11:36:03 +02:00
|
|
|
}
|
|
|
|
|
|
2021-07-14 16:02:25 +02:00
|
|
|
LauncherInterfacePrivate::~LauncherInterfacePrivate()
|
2021-07-07 11:36:03 +02:00
|
|
|
{
|
|
|
|
|
m_server->disconnect();
|
|
|
|
|
}
|
|
|
|
|
|
2021-07-14 16:02:25 +02:00
|
|
|
void LauncherInterfacePrivate::doStart()
|
2021-07-07 11:36:03 +02:00
|
|
|
{
|
|
|
|
|
if (++m_startRequests > 1)
|
|
|
|
|
return;
|
|
|
|
|
const QString &socketName = launcherSocketName();
|
|
|
|
|
QLocalServer::removeServer(socketName);
|
|
|
|
|
if (!m_server->listen(socketName)) {
|
|
|
|
|
emit errorOccurred(m_server->errorString());
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
m_process = new LauncherProcess(this);
|
2021-07-14 16:02:25 +02:00
|
|
|
connect(m_process, &QProcess::errorOccurred, this, &LauncherInterfacePrivate::handleProcessError);
|
2021-07-07 11:36:03 +02:00
|
|
|
connect(m_process,
|
|
|
|
|
static_cast<void (QProcess::*)(int, QProcess::ExitStatus)>(&QProcess::finished),
|
2021-07-14 16:02:25 +02:00
|
|
|
this, &LauncherInterfacePrivate::handleProcessFinished);
|
2021-07-07 11:36:03 +02:00
|
|
|
connect(m_process, &QProcess::readyReadStandardError,
|
2021-07-14 16:02:25 +02:00
|
|
|
this, &LauncherInterfacePrivate::handleProcessStderr);
|
2021-07-07 11:36:03 +02:00
|
|
|
m_process->start(qApp->applicationDirPath() + QLatin1Char('/')
|
|
|
|
|
+ QLatin1String(RELATIVE_LIBEXEC_PATH)
|
|
|
|
|
+ QLatin1String("/qtcreator_processlauncher"),
|
|
|
|
|
QStringList(m_server->fullServerName()));
|
|
|
|
|
}
|
|
|
|
|
|
2021-07-14 16:02:25 +02:00
|
|
|
void LauncherInterfacePrivate::doStop()
|
2021-07-07 11:36:03 +02:00
|
|
|
{
|
|
|
|
|
if (--m_startRequests > 0)
|
|
|
|
|
return;
|
|
|
|
|
m_server->close();
|
|
|
|
|
if (!m_process)
|
|
|
|
|
return;
|
|
|
|
|
m_process->disconnect();
|
|
|
|
|
m_socket->shutdown();
|
|
|
|
|
m_process->waitForFinished(3000);
|
|
|
|
|
m_process->deleteLater();
|
|
|
|
|
m_process = nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
2021-07-14 16:02:25 +02:00
|
|
|
void LauncherInterfacePrivate::handleNewConnection()
|
2021-07-07 11:36:03 +02:00
|
|
|
{
|
|
|
|
|
QLocalSocket * const socket = m_server->nextPendingConnection();
|
|
|
|
|
if (!socket)
|
|
|
|
|
return;
|
|
|
|
|
m_server->close();
|
|
|
|
|
m_socket->setSocket(socket);
|
|
|
|
|
}
|
|
|
|
|
|
2021-07-14 16:02:25 +02:00
|
|
|
void LauncherInterfacePrivate::handleProcessError()
|
2021-07-07 11:36:03 +02:00
|
|
|
{
|
|
|
|
|
if (m_process->error() == QProcess::FailedToStart) {
|
|
|
|
|
const QString launcherPathForUser
|
|
|
|
|
= QDir::toNativeSeparators(QDir::cleanPath(m_process->program()));
|
|
|
|
|
emit errorOccurred(QCoreApplication::translate("Utils::LauncherSocket",
|
|
|
|
|
"Failed to start process launcher at '%1': %2")
|
|
|
|
|
.arg(launcherPathForUser, m_process->errorString()));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-07-14 16:02:25 +02:00
|
|
|
void LauncherInterfacePrivate::handleProcessFinished()
|
2021-07-07 11:36:03 +02:00
|
|
|
{
|
|
|
|
|
emit errorOccurred(QCoreApplication::translate("Utils::LauncherSocket",
|
|
|
|
|
"Process launcher closed unexpectedly: %1")
|
|
|
|
|
.arg(m_process->errorString()));
|
|
|
|
|
}
|
|
|
|
|
|
2021-07-14 16:02:25 +02:00
|
|
|
void LauncherInterfacePrivate::handleProcessStderr()
|
2021-07-07 11:36:03 +02:00
|
|
|
{
|
|
|
|
|
qDebug() << "[launcher]" << m_process->readAllStandardError();
|
|
|
|
|
}
|
|
|
|
|
|
2021-07-14 16:02:25 +02:00
|
|
|
} // namespace Internal
|
|
|
|
|
|
|
|
|
|
using namespace Utils::Internal;
|
|
|
|
|
|
|
|
|
|
LauncherInterface::LauncherInterface()
|
|
|
|
|
: m_private(new LauncherInterfacePrivate())
|
|
|
|
|
{
|
|
|
|
|
m_private->moveToThread(&m_thread);
|
|
|
|
|
connect(m_private, &LauncherInterfacePrivate::errorOccurred,
|
|
|
|
|
this, &LauncherInterface::errorOccurred);
|
|
|
|
|
connect(&m_thread, &QThread::finished, m_private, &QObject::deleteLater);
|
|
|
|
|
m_thread.start();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
LauncherInterface &LauncherInterface::instance()
|
|
|
|
|
{
|
|
|
|
|
static LauncherInterface p;
|
|
|
|
|
return p;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
LauncherInterface::~LauncherInterface()
|
|
|
|
|
{
|
|
|
|
|
m_thread.quit();
|
|
|
|
|
m_thread.wait();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void LauncherInterface::startLauncher()
|
|
|
|
|
{
|
2021-08-18 10:45:57 +02:00
|
|
|
// Call in launcher's thread.
|
2021-07-14 16:02:25 +02:00
|
|
|
QMetaObject::invokeMethod(instance().m_private, &LauncherInterfacePrivate::doStart);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void LauncherInterface::stopLauncher()
|
|
|
|
|
{
|
2021-08-18 10:45:57 +02:00
|
|
|
// Call in launcher's thread.
|
2021-07-14 16:02:25 +02:00
|
|
|
QMetaObject::invokeMethod(instance().m_private, &LauncherInterfacePrivate::doStop);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Internal::LauncherSocket *LauncherInterface::socket()
|
|
|
|
|
{
|
|
|
|
|
return instance().m_private->socket();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2021-07-07 11:36:03 +02:00
|
|
|
} // namespace Utils
|
2021-07-14 16:02:25 +02:00
|
|
|
|
|
|
|
|
#include "launcherinterface.moc"
|