2018-10-18 11:57:19 +02:00
|
|
|
/****************************************************************************
|
|
|
|
|
**
|
|
|
|
|
** Copyright (C) 2018 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 "perfdatareader.h"
|
|
|
|
|
#include "perfprofilerconstants.h"
|
|
|
|
|
#include "perfprofilerruncontrol.h"
|
|
|
|
|
#include "perfprofilertool.h"
|
|
|
|
|
#include "perfrunconfigurationaspect.h"
|
|
|
|
|
#include "perfsettings.h"
|
|
|
|
|
|
|
|
|
|
#include <coreplugin/icore.h>
|
|
|
|
|
#include <coreplugin/messagemanager.h>
|
|
|
|
|
#include <projectexplorer/devicesupport/deviceprocess.h>
|
|
|
|
|
#include <projectexplorer/devicesupport/idevice.h>
|
|
|
|
|
#include <projectexplorer/kitinformation.h>
|
|
|
|
|
#include <projectexplorer/target.h>
|
|
|
|
|
#include <ssh/sshconnection.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()->runConfiguration());
|
|
|
|
|
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::DebugFormat);
|
|
|
|
|
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 runConfig = runControl->runConfiguration();
|
|
|
|
|
auto perfAspect = static_cast<PerfRunConfigurationAspect *>(runConfig->aspect(Constants::PerfSettingsId));
|
|
|
|
|
QTC_ASSERT(perfAspect, return);
|
|
|
|
|
PerfSettings *settings = static_cast<PerfSettings *>(perfAspect->currentSettings());
|
|
|
|
|
QTC_ASSERT(settings, return);
|
|
|
|
|
m_perfRecordArguments = settings->perfRecordArguments();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void start() override
|
|
|
|
|
{
|
|
|
|
|
m_process = device()->createProcess(this);
|
|
|
|
|
if (!m_process) {
|
|
|
|
|
reportFailure(tr("Could not start device process."));
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
connect(m_process, &DeviceProcess::started, this, &RunWorker::reportStarted);
|
|
|
|
|
connect(m_process, &DeviceProcess::finished, this, &RunWorker::reportStopped);
|
|
|
|
|
connect(m_process, &DeviceProcess::error, [this](QProcess::ProcessError e) {
|
|
|
|
|
// The terminate() below will frequently lead to QProcess::Crashed. We're not interested
|
|
|
|
|
// in that. FailedToStart is the only actual failure.
|
|
|
|
|
if (e == QProcess::FailedToStart) {
|
2019-03-01 16:58:35 +01:00
|
|
|
QString msg = tr("Perf Process Failed to Start");
|
2018-10-18 11:57:19 +02:00
|
|
|
QMessageBox::warning(Core::ICore::mainWindow(),
|
2019-03-01 16:58:35 +01:00
|
|
|
msg, tr("Make sure that you are running a recent Linux kernel and "
|
|
|
|
|
"that the \"perf\" utility is available."));
|
2018-10-18 11:57:19 +02:00
|
|
|
reportFailure(msg);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
Runnable perfRunnable = runnable();
|
|
|
|
|
|
|
|
|
|
QStringList arguments;
|
|
|
|
|
arguments << "record";
|
|
|
|
|
arguments += m_perfRecordArguments;
|
|
|
|
|
arguments << "-o" << "-" << "--" << perfRunnable.executable
|
|
|
|
|
<< Utils::QtcProcess::splitArgs(perfRunnable.commandLineArguments,
|
|
|
|
|
Utils::OsTypeLinux);
|
|
|
|
|
|
|
|
|
|
perfRunnable.executable = "perf";
|
|
|
|
|
perfRunnable.commandLineArguments = Utils::QtcProcess::joinArgs(arguments,
|
|
|
|
|
Utils::OsTypeLinux);
|
|
|
|
|
m_process->start(perfRunnable);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void stop() override
|
|
|
|
|
{
|
|
|
|
|
if (m_process)
|
|
|
|
|
m_process->terminate();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DeviceProcess *recorder() { return m_process; }
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
QPointer<DeviceProcess> 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 (auto perfRecorder = device()->workerCreator("PerfRecorder")) {
|
|
|
|
|
m_perfRecordWorker = perfRecorder(runControl);
|
|
|
|
|
|
|
|
|
|
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.
|
|
|
|
|
DeviceProcess *recorder = prw->recorder();
|
|
|
|
|
connect(recorder, &DeviceProcess::readyReadStandardError, this, [this, recorder] {
|
|
|
|
|
appendMessage(QString::fromLocal8Bit(recorder->readAllStandardError()),
|
|
|
|
|
Utils::StdErrFormat);
|
|
|
|
|
});
|
2019-01-07 14:30:39 +01:00
|
|
|
connect(recorder, &DeviceProcess::readyReadStandardOutput, this, [this, reader, recorder] {
|
|
|
|
|
if (!reader->feedParser(recorder->readAllStandardOutput()))
|
2019-03-01 16:58:35 +01:00
|
|
|
reportFailure(tr("Failed to transfer Perf data to perfparser."));
|
2018-10-18 11:57:19 +02:00
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
reportStarted();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} // namespace Internal
|
|
|
|
|
} // namespace PerfProfiler
|
|
|
|
|
|
|
|
|
|
#include "perfprofilerruncontrol.moc"
|