Files
qt-creator/src/plugins/remotelinux/killappstep.cpp

132 lines
3.6 KiB
C++
Raw Normal View History

// Copyright (C) 2022 The Qt Company Ltd.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0+ OR GPL-3.0 WITH Qt-GPL-exception-1.0
#include "killappstep.h"
#include "abstractremotelinuxdeploystep.h"
#include "abstractremotelinuxdeployservice.h"
#include "remotelinux_constants.h"
#include "remotelinuxtr.h"
#include <projectexplorer/devicesupport/idevice.h>
#include <projectexplorer/projectexplorerconstants.h>
#include <projectexplorer/runcontrol.h>
#include <projectexplorer/target.h>
#include <utils/qtcassert.h>
using namespace ProjectExplorer;
using namespace Utils;
namespace RemoteLinux::Internal {
class KillAppService : public AbstractRemoteLinuxDeployService
{
public:
~KillAppService() override;
void setRemoteExecutable(const QString &filePath);
private:
void handleStdErr();
void handleProcessFinished();
bool isDeploymentNecessary() const override;
void doDeploy() override;
void stopDeployment() override;
void handleSignalOpFinished(const QString &errorMessage);
void cleanup();
void finishDeployment();
QString m_remoteExecutable;
DeviceProcessSignalOperation::Ptr m_signalOperation;
};
KillAppService::~KillAppService()
{
cleanup();
}
void KillAppService::setRemoteExecutable(const QString &filePath)
{
m_remoteExecutable = filePath;
}
bool KillAppService::isDeploymentNecessary() const
{
return !m_remoteExecutable.isEmpty();
}
void KillAppService::doDeploy()
{
m_signalOperation = deviceConfiguration()->signalOperation();
if (!m_signalOperation) {
handleDeploymentDone();
return;
}
connect(m_signalOperation.data(), &DeviceProcessSignalOperation::finished,
this, &KillAppService::handleSignalOpFinished);
emit progressMessage(Tr::tr("Trying to kill \"%1\" on remote device...").arg(m_remoteExecutable));
m_signalOperation->killProcess(m_remoteExecutable);
}
void KillAppService::cleanup()
{
if (m_signalOperation) {
disconnect(m_signalOperation.data(), nullptr, this, nullptr);
m_signalOperation.clear();
}
}
void KillAppService::finishDeployment()
{
cleanup();
handleDeploymentDone();
}
void KillAppService::stopDeployment()
{
finishDeployment();
}
void KillAppService::handleSignalOpFinished(const QString &errorMessage)
{
if (errorMessage.isEmpty())
emit progressMessage(Tr::tr("Remote application killed."));
else
emit progressMessage(Tr::tr("Failed to kill remote application. Assuming it was not running."));
finishDeployment();
}
class KillAppStep : public AbstractRemoteLinuxDeployStep
{
public:
KillAppStep(BuildStepList *bsl, Id id) : AbstractRemoteLinuxDeployStep(bsl, id)
{
auto service = createDeployService<Internal::KillAppService>();
setWidgetExpandedByDefault(false);
setInternalInitializer([this, service] {
Target * const theTarget = target();
QTC_ASSERT(theTarget, return CheckResult::failure());
RunConfiguration * const rc = theTarget->activeRunConfiguration();
const QString remoteExe = rc ? rc->runnable().command.executable().path() : QString();
service->setRemoteExecutable(remoteExe);
return CheckResult::success();
});
}
};
KillAppStepFactory::KillAppStepFactory()
{
registerStep<KillAppStep>(Constants::KillAppStepId);
setDisplayName(Tr::tr("Kill current application instance"));
setSupportedConfiguration(RemoteLinux::Constants::DeployToGenericLinux);
setSupportedStepList(ProjectExplorer::Constants::BUILDSTEPS_DEPLOY);
}
} // RemoteLinux::Internal