2022-08-19 15:59:36 +02:00
|
|
|
// Copyright (C) 2021 The Qt Company Ltd.
|
|
|
|
|
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0+ OR GPL-3.0 WITH Qt-GPL-exception-1.0
|
2021-08-06 16:20:47 +02:00
|
|
|
|
|
|
|
|
#include "processutils.h"
|
|
|
|
|
|
2021-08-09 12:17:13 +02:00
|
|
|
#ifdef Q_OS_WIN
|
2022-03-14 17:02:11 +01:00
|
|
|
#ifdef QTCREATOR_PCH_H
|
|
|
|
|
#define CALLBACK WINAPI
|
|
|
|
|
#endif
|
2021-08-09 12:17:13 +02:00
|
|
|
#include <qt_windows.h>
|
2021-08-09 14:01:14 +02:00
|
|
|
#else
|
|
|
|
|
#include <errno.h>
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <unistd.h>
|
2021-08-09 12:17:13 +02:00
|
|
|
#endif
|
|
|
|
|
|
2021-08-06 16:20:47 +02:00
|
|
|
namespace Utils {
|
|
|
|
|
|
|
|
|
|
QIODevice::OpenMode ProcessStartHandler::openMode() const
|
|
|
|
|
{
|
|
|
|
|
if (m_processMode == ProcessMode::Writer)
|
|
|
|
|
return QIODevice::ReadWrite; // some writers also read data
|
|
|
|
|
if (m_writeData.isEmpty())
|
|
|
|
|
return QIODevice::ReadOnly; // only reading
|
|
|
|
|
return QIODevice::ReadWrite; // initial write and then reading (close the write channel)
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-16 10:43:58 +01:00
|
|
|
void ProcessStartHandler::handleProcessStart()
|
2021-08-06 16:20:47 +02:00
|
|
|
{
|
|
|
|
|
if (m_processMode == ProcessMode::Writer)
|
|
|
|
|
return;
|
|
|
|
|
if (m_writeData.isEmpty())
|
2021-12-16 10:43:58 +01:00
|
|
|
m_process->closeWriteChannel();
|
2021-08-06 16:20:47 +02:00
|
|
|
}
|
|
|
|
|
|
2021-12-16 10:43:58 +01:00
|
|
|
void ProcessStartHandler::handleProcessStarted()
|
2021-08-06 16:20:47 +02:00
|
|
|
{
|
|
|
|
|
if (!m_writeData.isEmpty()) {
|
2021-12-16 10:43:58 +01:00
|
|
|
m_process->write(m_writeData);
|
2021-08-06 16:20:47 +02:00
|
|
|
m_writeData = {};
|
|
|
|
|
if (m_processMode == ProcessMode::Reader)
|
2021-12-16 10:43:58 +01:00
|
|
|
m_process->closeWriteChannel();
|
2021-08-06 16:20:47 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-16 10:43:58 +01:00
|
|
|
void ProcessStartHandler::setBelowNormalPriority()
|
2021-08-09 12:17:13 +02:00
|
|
|
{
|
|
|
|
|
#ifdef Q_OS_WIN
|
2021-12-16 10:43:58 +01:00
|
|
|
m_process->setCreateProcessArgumentsModifier(
|
2021-08-09 12:17:13 +02:00
|
|
|
[](QProcess::CreateProcessArguments *args) {
|
|
|
|
|
args->flags |= BELOW_NORMAL_PRIORITY_CLASS;
|
|
|
|
|
});
|
|
|
|
|
#endif // Q_OS_WIN
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-16 10:43:58 +01:00
|
|
|
void ProcessStartHandler::setNativeArguments(const QString &arguments)
|
2021-08-09 12:57:22 +02:00
|
|
|
{
|
|
|
|
|
#ifdef Q_OS_WIN
|
|
|
|
|
if (!arguments.isEmpty())
|
2021-12-16 10:43:58 +01:00
|
|
|
m_process->setNativeArguments(arguments);
|
2021-08-09 12:57:22 +02:00
|
|
|
#else
|
|
|
|
|
Q_UNUSED(arguments)
|
|
|
|
|
#endif // Q_OS_WIN
|
|
|
|
|
}
|
2021-08-09 12:17:13 +02:00
|
|
|
|
2022-03-14 17:02:11 +01:00
|
|
|
#ifdef Q_OS_WIN
|
|
|
|
|
static BOOL sendMessage(UINT message, HWND hwnd, LPARAM lParam)
|
|
|
|
|
{
|
|
|
|
|
DWORD dwProcessID;
|
|
|
|
|
GetWindowThreadProcessId(hwnd, &dwProcessID);
|
|
|
|
|
if ((DWORD)lParam == dwProcessID) {
|
|
|
|
|
SendNotifyMessage(hwnd, message, 0, 0);
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
return TRUE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
BOOL CALLBACK sendShutDownMessageToAllWindowsOfProcess_enumWnd(HWND hwnd, LPARAM lParam)
|
|
|
|
|
{
|
|
|
|
|
static UINT uiShutDownMessage = RegisterWindowMessage(L"qtcctrlcstub_shutdown");
|
|
|
|
|
return sendMessage(uiShutDownMessage, hwnd, lParam);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
BOOL CALLBACK sendInterruptMessageToAllWindowsOfProcess_enumWnd(HWND hwnd, LPARAM lParam)
|
|
|
|
|
{
|
|
|
|
|
static UINT uiInterruptMessage = RegisterWindowMessage(L"qtcctrlcstub_interrupt");
|
|
|
|
|
return sendMessage(uiInterruptMessage, hwnd, lParam);
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
2022-07-14 15:51:24 +02:00
|
|
|
ProcessHelper::ProcessHelper(QObject *parent)
|
|
|
|
|
: QProcess(parent), m_processStartHandler(this)
|
|
|
|
|
{
|
|
|
|
|
#if defined(Q_OS_UNIX)
|
|
|
|
|
setChildProcessModifier([this] { setupChildProcess_impl(); });
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2022-03-14 17:02:11 +01:00
|
|
|
void ProcessHelper::setUseCtrlCStub(bool enabled)
|
|
|
|
|
{
|
|
|
|
|
m_useCtrlCStub = enabled;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ProcessHelper::terminateProcess()
|
|
|
|
|
{
|
|
|
|
|
#ifdef Q_OS_WIN
|
|
|
|
|
if (m_useCtrlCStub)
|
|
|
|
|
EnumWindows(sendShutDownMessageToAllWindowsOfProcess_enumWnd, processId());
|
|
|
|
|
else
|
|
|
|
|
terminate();
|
|
|
|
|
#else
|
|
|
|
|
terminate();
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ProcessHelper::terminateProcess(QProcess *process)
|
|
|
|
|
{
|
|
|
|
|
ProcessHelper *helper = qobject_cast<ProcessHelper *>(process);
|
|
|
|
|
if (helper)
|
|
|
|
|
helper->terminateProcess();
|
|
|
|
|
else
|
|
|
|
|
process->terminate();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ProcessHelper::interruptPid(qint64 pid)
|
|
|
|
|
{
|
|
|
|
|
#ifdef Q_OS_WIN
|
|
|
|
|
EnumWindows(sendInterruptMessageToAllWindowsOfProcess_enumWnd, pid);
|
|
|
|
|
#else
|
|
|
|
|
Q_UNUSED(pid)
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ProcessHelper::interruptProcess(QProcess *process)
|
|
|
|
|
{
|
|
|
|
|
ProcessHelper *helper = qobject_cast<ProcessHelper *>(process);
|
|
|
|
|
if (helper && helper->m_useCtrlCStub)
|
|
|
|
|
ProcessHelper::interruptPid(process->processId());
|
|
|
|
|
}
|
2021-08-09 14:01:14 +02:00
|
|
|
|
|
|
|
|
void ProcessHelper::setupChildProcess_impl()
|
|
|
|
|
{
|
|
|
|
|
#if defined Q_OS_UNIX
|
|
|
|
|
// nice value range is -20 to +19 where -20 is highest, 0 default and +19 is lowest
|
|
|
|
|
if (m_lowPriority) {
|
|
|
|
|
errno = 0;
|
|
|
|
|
if (::nice(5) == -1 && errno != 0)
|
|
|
|
|
perror("Failed to set nice value");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Disable terminal by becoming a session leader.
|
|
|
|
|
if (m_unixTerminalDisabled)
|
|
|
|
|
setsid();
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-06 16:20:47 +02:00
|
|
|
} // namespace Utils
|