1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180
|
/*
SPDX-FileCopyrightText: 2017 Anton Anikin <anton.anikin@htower.ru>
SPDX-License-Identifier: GPL-2.0-or-later
*/
#include "plugin.h"
#include "config/globalconfigpage.h"
#include "debug.h"
#include "job.h"
#include "utils.h"
#include "visualizer.h"
#include <config-kdevelop.h>
#if HAVE_KSYSGUARD
#include "dialogs/processselection.h"
#include <QPointer>
#endif
#include <execute/iexecuteplugin.h>
#include <interfaces/iplugincontroller.h>
#include <interfaces/iuicontroller.h>
#include <interfaces/launchconfigurationtype.h>
#include <shell/core.h>
#include <shell/launchconfiguration.h>
#include <shell/runcontroller.h>
#include <sublime/message.h>
#include <util/executecompositejob.h>
// KF
#include <KActionCollection>
#include <KPluginFactory>
// Qt
#include <QAction>
#include <QApplication>
#include <QFile>
K_PLUGIN_FACTORY_WITH_JSON(HeaptrackFactory, "kdevheaptrack.json", registerPlugin<Heaptrack::Plugin>();)
namespace {
void postErrorMessage(const QString& messageText)
{
auto* const message = new Sublime::Message(messageText, Sublime::Message::Error);
KDevelop::ICore::self()->uiController()->postMessage(message);
}
}
namespace Heaptrack
{
Plugin::Plugin(QObject* parent, const KPluginMetaData& metaData, const QVariantList&)
: IPlugin(QStringLiteral("kdevheaptrack"), parent, metaData)
{
setXMLFile(QStringLiteral("kdevheaptrack.rc"));
m_launchAction = new QAction(
QIcon::fromTheme(QStringLiteral("office-chart-area")),
i18nc("@action", "Run Heaptrack Analysis"),
this);
connect(m_launchAction, &QAction::triggered, this, &Plugin::launchHeaptrack);
actionCollection()->addAction(QStringLiteral("heaptrack_launch"), m_launchAction);
#if HAVE_KSYSGUARD
m_attachAction = new QAction(
QIcon::fromTheme(QStringLiteral("office-chart-area")),
i18nc("@action", "Attach to Process with Heaptrack"),
this);
connect(m_attachAction, &QAction::triggered, this, &Plugin::attachHeaptrack);
actionCollection()->addAction(QStringLiteral("heaptrack_attach"), m_attachAction);
#endif
}
Plugin::~Plugin()
{
}
void Plugin::launchHeaptrack()
{
IExecutePlugin* executePlugin = nullptr;
// First we should check that our "kdevexecute" plugin is loaded. This is needed since
// current plugin controller logic allows us to unload this plugin with keeping dependent
// plugins like Heaptrack in "loaded" state. This seems to be wrong behaviour but now we have
// to do additional checks.
// TODO fix plugin controller to avoid such inconsistent states.
auto pluginController = core()->pluginController();
if (auto plugin = pluginController->pluginForExtension(
QStringLiteral("org.kdevelop.IExecutePlugin"), QStringLiteral("kdevexecute"))) {
executePlugin = plugin->extension<IExecutePlugin>();
} else {
auto pluginInfo = pluginController->infoForPluginId(QStringLiteral("kdevexecute"));
postErrorMessage(i18n("Unable to start Heaptrack analysis - \"%1\" plugin is not loaded.", pluginInfo.name()));
return;
}
auto runController = KDevelop::Core::self()->runControllerInternal();
auto defaultLaunch = runController->defaultLaunch();
if (!defaultLaunch) {
runController->showConfigurationDialog();
defaultLaunch = runController->defaultLaunch();
if (!defaultLaunch) {
postErrorMessage(i18n("Configure a native application launch to perform Heaptrack analysis on."));
return;
}
}
if (!defaultLaunch->type()->launcherForId(QStringLiteral("nativeAppLauncher"))) {
postErrorMessage(i18n("Heaptrack analysis can be started only for native applications."));
return;
}
auto heaptrackJob = new Job(defaultLaunch, executePlugin);
connect(heaptrackJob, &Job::finished, this, &Plugin::jobFinished);
QList<KJob*> jobList;
if (KJob* depJob = executePlugin->dependencyJob(defaultLaunch)) {
jobList += depJob;
}
jobList += heaptrackJob;
auto ecJob = new KDevelop::ExecuteCompositeJob(runController, jobList);
ecJob->setObjectName(heaptrackJob->statusName());
runController->registerJob(ecJob);
m_launchAction->setEnabled(false);
}
void Plugin::attachHeaptrack()
{
#if HAVE_KSYSGUARD
const auto pid = KDevMI::askUserForProcessId(activeMainWindow());
if (pid == 0) {
return;
}
auto heaptrackJob = new Job(pid);
connect(heaptrackJob, &Job::finished, this, &Plugin::jobFinished);
heaptrackJob->setObjectName(heaptrackJob->statusName());
core()->runController()->registerJob(heaptrackJob);
m_launchAction->setEnabled(false);
#endif
}
void Plugin::jobFinished(KJob* kjob)
{
auto job = static_cast<Job*>(kjob);
Q_ASSERT(job);
if (job->status() == KDevelop::OutputExecuteJob::JobStatus::JobSucceeded) {
auto visualizer = new Visualizer(job->resultsFile(), this);
visualizer->start();
} else {
QFile::remove(job->resultsFile());
}
m_launchAction->setEnabled(true);
}
int Plugin::configPages() const
{
return 1;
}
KDevelop::ConfigPage* Plugin::configPage(int number, QWidget* parent)
{
if (number) {
return nullptr;
}
return new GlobalConfigPage(this, parent);
}
}
#include "plugin.moc"
#include "moc_plugin.cpp"
|