/**************************************************************************** ** ** Copyright (C) 2016 The Qt Company Ltd. ** Author: Milian Wolff, KDAB (milian.wolff@kdab.com) ** 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 "valgrindplugin.h" #include "callgrindtool.h" #include "memchecktool.h" #include "memcheckengine.h" #include "valgrindruncontrolfactory.h" #include "valgrindsettings.h" #include "valgrindconfigwidget.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace Analyzer; using namespace Core; using namespace ProjectExplorer; namespace Valgrind { namespace Internal { static ValgrindGlobalSettings *theGlobalSettings = 0; class ValgrindOptionsPage : public IOptionsPage { public: explicit ValgrindOptionsPage() { setId(ANALYZER_VALGRIND_SETTINGS); setDisplayName(QCoreApplication::translate("Valgrind::Internal::ValgrindOptionsPage", "Valgrind")); setCategory("T.Analyzer"); setDisplayCategory(QCoreApplication::translate("Analyzer", "Analyzer")); setCategoryIcon(QLatin1String(":/images/analyzer_category.png")); } QWidget *widget() { if (!m_widget) m_widget = new ValgrindConfigWidget(theGlobalSettings, 0, true); return m_widget; } void apply() { theGlobalSettings->writeSettings(); } void finish() { delete m_widget; } private: QPointer m_widget; }; ValgrindPlugin::~ValgrindPlugin() { delete theGlobalSettings; theGlobalSettings = 0; } bool ValgrindPlugin::initialize(const QStringList &, QString *) { theGlobalSettings = new ValgrindGlobalSettings(); theGlobalSettings->readSettings(); addAutoReleasedObject(new ValgrindOptionsPage()); addAutoReleasedObject(new ValgrindRunControlFactory()); return true; } void ValgrindPlugin::extensionsInitialized() { using namespace std::placeholders; AnalyzerAction *action = 0; QString callgrindToolTip = tr("Valgrind Function Profile uses the " "Callgrind tool to record function calls when a program runs."); QString memcheckToolTip = tr("Valgrind Analyze Memory uses the " "Memcheck tool to find memory leaks."); auto mcTool = new MemcheckTool(this); auto mcWidgetCreator = [mcTool] { return mcTool->createWidgets(); }; auto cgTool = new CallgrindTool(this); auto cgWidgetCreator = [cgTool] { return cgTool->createWidgets(); }; auto cgRunControlCreator = [cgTool](const AnalyzerStartParameters &sp, RunConfiguration *runConfiguration, Core::Id) { return cgTool->createRunControl(sp, runConfiguration); }; if (!Utils::HostOsInfo::isWindowsHost()) { action = new AnalyzerAction(this); action->setActionId("Memcheck.Local"); action->setToolId("Memcheck"); action->setWidgetCreator(mcWidgetCreator); action->setRunControlCreator(std::bind(&MemcheckTool::createRunControl, mcTool, _1, _2, MEMCHECK_RUN_MODE)); action->setToolMode(DebugMode); action->setRunMode(MEMCHECK_RUN_MODE); action->setText(tr("Valgrind Memory Analyzer")); action->setToolTip(memcheckToolTip); action->setMenuGroup(Analyzer::Constants::G_ANALYZER_TOOLS); action->setEnabled(false); AnalyzerManager::addAction(action); auto mcgTool = new MemcheckTool(this); action = new AnalyzerAction(this); action->setActionId("MemcheckWithGdb.Local"); action->setToolId("MemcheckWithGdb"); action->setWidgetCreator([mcgTool] { return mcgTool->createWidgets(); }); action->setRunControlCreator(std::bind(&MemcheckTool::createRunControl, mcgTool, _1, _2, MEMCHECK_WITH_GDB_RUN_MODE)); action->setToolMode(DebugMode); action->setRunMode(MEMCHECK_WITH_GDB_RUN_MODE); action->setText(tr("Valgrind Memory Analyzer with GDB")); action->setToolTip(tr("Valgrind Analyze Memory with GDB uses the " "Memcheck tool to find memory leaks.\nWhen a problem is detected, " "the application is interrupted and can be debugged.")); action->setMenuGroup(Analyzer::Constants::G_ANALYZER_TOOLS); action->setEnabled(false); AnalyzerManager::addAction(action); action = new AnalyzerAction(this); action->setActionId(CallgrindLocalActionId); action->setToolId(CallgrindToolId); action->setWidgetCreator(cgWidgetCreator); action->setRunControlCreator(cgRunControlCreator); action->setToolMode(OptimizedMode); action->setRunMode(CALLGRIND_RUN_MODE); action->setText(tr("Valgrind Function Profiler")); action->setToolTip(callgrindToolTip); action->setMenuGroup(Analyzer::Constants::G_ANALYZER_TOOLS); action->setEnabled(false); AnalyzerManager::addAction(action); } action = new AnalyzerAction(this); action->setActionId("Memcheck.Remote"); action->setToolId("Memcheck"); action->setWidgetCreator(mcWidgetCreator); action->setCustomToolStarter([mcTool] { StartRemoteDialog dlg; if (dlg.exec() != QDialog::Accepted) return; AnalyzerStartParameters sp; sp.connParams = dlg.sshParams(); sp.debuggee = dlg.executable(); sp.debuggeeArgs = dlg.arguments(); ValgrindRunControl *rc = mcTool->createRunControl(sp, 0, MEMCHECK_RUN_MODE); QTC_ASSERT(rc, return); rc->setDisplayName(dlg.executable()); rc->setWorkingDirectory(dlg.workingDirectory()); rc->setCustomStart(); ProjectExplorerPlugin::startRunControl(rc, MEMCHECK_RUN_MODE); }); action->setText(tr("Valgrind Memory Analyzer (External Remote Application)")); action->setToolTip(memcheckToolTip); action->setMenuGroup(Analyzer::Constants::G_ANALYZER_REMOTE_TOOLS); AnalyzerManager::addAction(action); action = new AnalyzerAction(this); action->setActionId(CallgrindRemoteActionId); action->setToolId(CallgrindToolId); action->setWidgetCreator(cgWidgetCreator); action->setCustomToolStarter([cgTool] { StartRemoteDialog dlg; if (dlg.exec() != QDialog::Accepted) return; AnalyzerStartParameters sp; sp.connParams = dlg.sshParams(); sp.debuggee = dlg.executable(); sp.debuggeeArgs = dlg.arguments(); ValgrindRunControl *rc = cgTool->createRunControl(sp, 0); QTC_ASSERT(rc, return); rc->setDisplayName(dlg.executable()); rc->setWorkingDirectory(dlg.workingDirectory()); rc->setCustomStart(); ProjectExplorerPlugin::startRunControl(rc, CALLGRIND_RUN_MODE); }); action->setText(tr("Valgrind Function Profiler (External Remote Application)")); action->setToolTip(callgrindToolTip); action->setMenuGroup(Analyzer::Constants::G_ANALYZER_REMOTE_TOOLS); AnalyzerManager::addAction(action); // If there is a CppEditor context menu add our own context menu actions. if (ActionContainer *editorContextMenu = ActionManager::actionContainer(CppEditor::Constants::M_CONTEXT)) { Context analyzerContext = Context(Analyzer::Constants::C_ANALYZEMODE); editorContextMenu->addSeparator(analyzerContext); QAction *action = new QAction(tr("Profile Costs of This Function and Its Callees"), this); action->setIcon(Analyzer::Icons::ANALYZER_CONTROL_START.icon()); connect(action, &QAction::triggered, cgTool, &CallgrindTool::handleShowCostsOfFunction); Command *cmd = ActionManager::registerAction(action, "Analyzer.Callgrind.ShowCostsOfFunction", analyzerContext); editorContextMenu->addAction(cmd); cmd->setAttribute(Command::CA_Hide); cmd->setAttribute(Command::CA_NonConfigurable); } } ValgrindGlobalSettings *ValgrindPlugin::globalSettings() { return theGlobalSettings; } } // namespace Internal } // namespace Valgrind