Files
qt-creator/src/plugins/perfprofiler/perfprofilerruncontrol.cpp
Alessandro Portale 88489fa956 PerfProfiler: Convert to using Tr::tr
Change-Id: Ied3f0a499a1b5c6da752667a44525ecce66febac
Reviewed-by: <github-actions-qt-creator@cristianadam.eu>
Reviewed-by: Alessandro Portale <alessandro.portale@qt.io>
2022-09-22 07:33:20 +00:00

206 lines
7.0 KiB
C++

// Copyright (C) 2018 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 "perfprofilerruncontrol.h"
#include "perfdatareader.h"
#include "perfprofilertool.h"
#include "perfrunconfigurationaspect.h"
#include "perfsettings.h"
#include <coreplugin/icore.h>
#include <coreplugin/messagemanager.h>
#include <projectexplorer/devicesupport/idevice.h>
#include <projectexplorer/kitinformation.h>
#include <projectexplorer/target.h>
#include <utils/qtcprocess.h>
#include <QAction>
#include <QMessageBox>
#include <QTcpServer>
using namespace ProjectExplorer;
using namespace Utils;
namespace PerfProfiler {
namespace Internal {
class PerfParserWorker : public RunWorker
{
Q_OBJECT
public:
PerfParserWorker(RunControl *runControl)
: RunWorker(runControl)
{
setId("PerfParser");
auto tool = PerfProfilerTool::instance();
m_reader.setTraceManager(tool->traceManager());
m_reader.triggerRecordingStateChange(tool->isRecording());
connect(tool, &PerfProfilerTool::recordingChanged,
&m_reader, &PerfDataReader::triggerRecordingStateChange);
connect(&m_reader, &PerfDataReader::updateTimestamps,
tool, &PerfProfilerTool::updateTime);
connect(&m_reader, &PerfDataReader::starting,
tool, &PerfProfilerTool::startLoading);
connect(&m_reader, &PerfDataReader::started, tool, &PerfProfilerTool::onReaderStarted);
connect(&m_reader, &PerfDataReader::finishing, this, [tool] {
// Temporarily disable buttons.
tool->setToolActionsEnabled(false);
});
connect(&m_reader, &PerfDataReader::finished, tool, &PerfProfilerTool::onReaderFinished);
connect(&m_reader, &PerfDataReader::processStarted, this, &RunWorker::reportStarted);
connect(&m_reader, &PerfDataReader::processFinished, this, &RunWorker::reportStopped);
connect(&m_reader, &PerfDataReader::processFailed, this, &RunWorker::reportFailure);
}
void start() override
{
QStringList args = m_reader.findTargetArguments(runControl());
QUrl url = runControl()->property("PerfConnection").toUrl();
if (url.isValid()) {
args.append(QStringList{"--host", url.host(), "--port", QString::number(url.port())});
}
appendMessage("PerfParser args: " + args.join(' '), Utils::NormalMessageFormat);
m_reader.createParser(args);
m_reader.startParser();
}
void stop() override
{
m_reader.stopParser();
}
PerfDataReader *reader() { return &m_reader;}
private:
PerfDataReader m_reader;
};
class LocalPerfRecordWorker : public RunWorker
{
Q_OBJECT
public:
LocalPerfRecordWorker(RunControl *runControl)
: RunWorker(runControl)
{
setId("LocalPerfRecordWorker");
auto perfAspect = runControl->aspect<PerfRunConfigurationAspect>();
QTC_ASSERT(perfAspect, return);
PerfSettings *settings = static_cast<PerfSettings *>(perfAspect->currentSettings);
QTC_ASSERT(settings, return);
m_perfRecordArguments = settings->perfRecordArguments();
}
void start() override
{
m_process = new QtcProcess(this);
connect(m_process, &QtcProcess::started, this, &RunWorker::reportStarted);
connect(m_process, &QtcProcess::done, this, [this] {
// The terminate() below will frequently lead to QProcess::Crashed. We're not interested
// in that. FailedToStart is the only actual failure.
if (m_process->error() == QProcess::FailedToStart) {
const QString msg = Tr::tr("Perf Process Failed to Start");
QMessageBox::warning(Core::ICore::dialogParent(), msg,
Tr::tr("Make sure that you are running a recent Linux kernel "
"and that the \"perf\" utility is available."));
reportFailure(msg);
return;
}
reportStopped();
});
CommandLine cmd({device()->filePath("perf"), {"record"}});
cmd.addArgs(m_perfRecordArguments);
cmd.addArgs({"-o", "-", "--"});
cmd.addCommandLineAsArgs(runControl()->commandLine(), CommandLine::Raw);
m_process->setCommand(cmd);
m_process->start();
}
void stop() override
{
if (m_process)
m_process->terminate();
}
QtcProcess *recorder() { return m_process; }
private:
QPointer<QtcProcess> m_process;
QStringList m_perfRecordArguments;
};
PerfProfilerRunner::PerfProfilerRunner(RunControl *runControl)
: RunWorker(runControl)
{
setId("PerfProfilerRunner");
m_perfParserWorker = new PerfParserWorker(runControl);
addStopDependency(m_perfParserWorker);
// If the parser is gone, there is no point in going on.
m_perfParserWorker->setEssential(true);
if ((m_perfRecordWorker = runControl->createWorker("PerfRecorder"))) {
m_perfParserWorker->addStartDependency(m_perfRecordWorker);
addStartDependency(m_perfParserWorker);
} else {
m_perfRecordWorker = new LocalPerfRecordWorker(runControl);
m_perfRecordWorker->addStartDependency(m_perfParserWorker);
addStartDependency(m_perfRecordWorker);
// In the local case, the parser won't automatically stop when the recorder does. So we need
// to mark the recorder as essential, too.
m_perfRecordWorker->setEssential(true);
}
m_perfParserWorker->addStopDependency(m_perfRecordWorker);
PerfProfilerTool::instance()->onWorkerCreation(runControl);
}
void PerfProfilerRunner::start()
{
auto tool = PerfProfilerTool::instance();
connect(tool->stopAction(), &QAction::triggered, runControl(), &RunControl::initiateStop);
connect(runControl(), &RunControl::started, PerfProfilerTool::instance(),
&PerfProfilerTool::onRunControlStarted);
connect(runControl(), &RunControl::stopped, PerfProfilerTool::instance(),
&PerfProfilerTool::onRunControlFinished);
connect(runControl(), &RunControl::finished, PerfProfilerTool::instance(),
&PerfProfilerTool::onRunControlFinished);
PerfDataReader *reader = m_perfParserWorker->reader();
if (auto prw = qobject_cast<LocalPerfRecordWorker *>(m_perfRecordWorker)) {
// That's the local case.
QtcProcess *recorder = prw->recorder();
connect(recorder, &QtcProcess::readyReadStandardError, this, [this, recorder] {
appendMessage(QString::fromLocal8Bit(recorder->readAllStandardError()),
Utils::StdErrFormat);
});
connect(recorder, &QtcProcess::readyReadStandardOutput, this, [this, reader, recorder] {
if (!reader->feedParser(recorder->readAllStandardOutput()))
reportFailure(Tr::tr("Failed to transfer Perf data to perfparser."));
});
}
reportStarted();
}
} // namespace Internal
} // namespace PerfProfiler
#include "perfprofilerruncontrol.moc"