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"
|
|
|
|
|
|
2021-08-19 14:01:40 +02:00
|
|
|
#include "filepath.h"
|
2021-07-07 11:36:03 +02:00
|
|
|
#include "launcherpackets.h"
|
|
|
|
|
#include "launchersocket.h"
|
2021-08-19 14:01:40 +02:00
|
|
|
#include "qtcassert.h"
|
2021-07-07 11:36:03 +02:00
|
|
|
|
2021-08-19 16:39:21 +02:00
|
|
|
#include <QCoreApplication>
|
|
|
|
|
#include <QDebug>
|
|
|
|
|
#include <QDir>
|
|
|
|
|
#include <QLocalServer>
|
|
|
|
|
#include <QProcess>
|
2021-07-07 11:36:03 +02:00
|
|
|
|
|
|
|
|
#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; }
|
|
|
|
|
|
2021-08-13 13:30:43 +02:00
|
|
|
void setPathToLauncher(const QString &path) { if (!path.isEmpty()) m_pathToLauncher = path; }
|
2021-08-19 14:01:40 +02:00
|
|
|
QString launcherFilePath() const { return m_pathToLauncher + QLatin1String("/qtcreator_processlauncher"); }
|
2021-07-14 16:02:25 +02:00
|
|
|
signals:
|
|
|
|
|
void errorOccurred(const QString &error);
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
QLocalServer * const m_server;
|
|
|
|
|
Internal::LauncherSocket *const m_socket;
|
|
|
|
|
Internal::LauncherProcess *m_process = nullptr;
|
2021-08-13 13:30:43 +02:00
|
|
|
QString m_pathToLauncher;
|
2021-07-14 16:02:25 +02:00
|
|
|
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-08-13 13:30:43 +02:00
|
|
|
m_pathToLauncher = qApp->applicationDirPath() + '/' + QLatin1String(RELATIVE_LIBEXEC_PATH);
|
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
|
|
|
{
|
|
|
|
|
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-08-19 14:01:40 +02:00
|
|
|
m_process->start(launcherFilePath(), QStringList(m_server->fullServerName()));
|
2021-07-07 11:36:03 +02:00
|
|
|
}
|
|
|
|
|
|
2021-07-14 16:02:25 +02:00
|
|
|
void LauncherInterfacePrivate::doStop()
|
2021-07-07 11:36:03 +02:00
|
|
|
{
|
|
|
|
|
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;
|
|
|
|
|
|
2021-08-24 15:41:16 +02:00
|
|
|
static LauncherInterface *s_instance = nullptr;
|
|
|
|
|
|
2021-07-14 16:02:25 +02:00
|
|
|
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()
|
|
|
|
|
{
|
|
|
|
|
m_thread.quit();
|
|
|
|
|
m_thread.wait();
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-19 14:01:40 +02:00
|
|
|
// Called from main thread
|
2021-08-13 13:30:43 +02:00
|
|
|
void LauncherInterface::startLauncher(const QString &pathToLauncher)
|
2021-07-14 16:02:25 +02:00
|
|
|
{
|
2021-08-24 15:41:16 +02:00
|
|
|
QTC_ASSERT(s_instance == nullptr, return);
|
|
|
|
|
s_instance = new LauncherInterface();
|
|
|
|
|
LauncherInterfacePrivate *p = s_instance->m_private;
|
2021-08-19 14:34:46 +02:00
|
|
|
p->setPathToLauncher(pathToLauncher);
|
|
|
|
|
const FilePath launcherFilePath = FilePath::fromString(p->launcherFilePath())
|
2021-08-19 14:01:40 +02:00
|
|
|
.cleanPath().withExecutableSuffix();
|
|
|
|
|
auto launcherIsNotExecutable = [&launcherFilePath]() {
|
|
|
|
|
qWarning() << "The Creator's process launcher"
|
|
|
|
|
<< launcherFilePath << "is not executable.";
|
|
|
|
|
};
|
2021-08-19 14:34:46 +02:00
|
|
|
QTC_ASSERT(launcherFilePath.isExecutableFile(), launcherIsNotExecutable(); return);
|
2021-08-18 10:45:57 +02:00
|
|
|
// Call in launcher's thread.
|
2021-08-19 14:34:46 +02:00
|
|
|
QMetaObject::invokeMethod(p, &LauncherInterfacePrivate::doStart);
|
2021-07-14 16:02:25 +02:00
|
|
|
}
|
|
|
|
|
|
2021-08-19 14:01:40 +02:00
|
|
|
// Called from main thread
|
2021-07-14 16:02:25 +02:00
|
|
|
void LauncherInterface::stopLauncher()
|
|
|
|
|
{
|
2021-08-24 15:41:16 +02:00
|
|
|
QTC_ASSERT(s_instance != nullptr, return);
|
|
|
|
|
LauncherInterfacePrivate *p = s_instance->m_private;
|
2021-08-18 10:45:57 +02:00
|
|
|
// Call in launcher's thread.
|
2021-08-25 14:59:00 +02:00
|
|
|
QMetaObject::invokeMethod(p, &LauncherInterfacePrivate::doStop, Qt::BlockingQueuedConnection);
|
2021-08-24 15:41:16 +02:00
|
|
|
delete s_instance;
|
|
|
|
|
s_instance = nullptr;
|
2021-07-14 16:02:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Internal::LauncherSocket *LauncherInterface::socket()
|
|
|
|
|
{
|
2021-08-24 15:41:16 +02:00
|
|
|
QTC_ASSERT(s_instance != nullptr, return nullptr);
|
|
|
|
|
return s_instance->m_private->socket();
|
2021-07-14 16:02:25 +02:00
|
|
|
}
|
|
|
|
|
|
2021-08-19 14:34:46 +02:00
|
|
|
bool LauncherInterface::isStarted()
|
|
|
|
|
{
|
2021-08-24 15:41:16 +02:00
|
|
|
return s_instance != nullptr;
|
2021-08-19 14:34:46 +02:00
|
|
|
}
|
2021-07-14 16:02:25 +02:00
|
|
|
|
2021-07-07 11:36:03 +02:00
|
|
|
} // namespace Utils
|
2021-07-14 16:02:25 +02:00
|
|
|
|
|
|
|
|
#include "launcherinterface.moc"
|