/**************************************************************************** ** ** Copyright (C) 2016 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 "callgrindtool.h" #include "callgrindcostdelegate.h" #include "callgrindcostview.h" #include "callgrindengine.h" #include "callgrindtextmark.h" #include "callgrindvisualisation.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace Debugger; using namespace Core; using namespace Valgrind::Callgrind; using namespace TextEditor; using namespace ProjectExplorer; using namespace Utils; namespace Valgrind { namespace Internal { const char CallgrindLocalActionId[] = "Callgrind.Local.Action"; const char CallgrindRemoteActionId[] = "Callgrind.Remote.Action"; const char CALLGRIND_RUN_MODE[] = "CallgrindTool.CallgrindRunMode"; class CallgrindToolPrivate : public QObject { Q_OBJECT public: CallgrindToolPrivate(); ~CallgrindToolPrivate() override; void setupRunner(CallgrindToolRunner *runner); void setParseData(ParseData *data); CostDelegate::CostFormat costFormat() const; void doClear(bool clearParseData); void updateEventCombo(); signals: void dumpRequested(); void resetRequested(); void pauseToggled(bool checked); public: void slotRequestDump(); void loadExternalLogFile(); void selectFunction(const Function *); void setCostFormat(CostDelegate::CostFormat format); void enableCycleDetection(bool enabled); void shortenTemplates(bool enabled); void setCostEvent(int index); /// This function will add custom text marks to the editor /// \note Call this after the data model has been populated void createTextMarks(); /// This function will clear all text marks from the editor void clearTextMarks(); void updateFilterString(); void updateCostFormat(); void handleFilterProjectCosts(); void handleShowCostsOfFunction(); void stackBrowserChanged(); /// If \param busy is true, all widgets get a busy cursor when hovered void setBusyCursor(bool busy); void dataFunctionSelected(const QModelIndex &index); void calleeFunctionSelected(const QModelIndex &index); void callerFunctionSelected(const QModelIndex &index); void visualisationFunctionSelected(const Function *function); void showParserResults(const ParseData *data); void takeParserDataFromRunControl(CallgrindToolRunner *rc); void takeParserData(ParseData *data); void engineFinished(); void editorOpened(IEditor *); void requestContextMenu(TextEditorWidget *widget, int line, QMenu *menu); void updateRunActions(); public: DataModel m_dataModel; DataProxyModel m_proxyModel; StackBrowser m_stackBrowser; CallModel m_callersModel; CallModel m_calleesModel; QSortFilterProxyModel m_callersProxy; QSortFilterProxyModel m_calleesProxy; // Callgrind widgets QPointer m_flatView; QPointer m_callersView; QPointer m_calleesView; QPointer m_visualization; QString m_lastFileName; // Navigation QAction *m_goBack = nullptr; QAction *m_goNext = nullptr; QLineEdit *m_searchFilter = nullptr; // Cost formatting QAction *m_filterProjectCosts = nullptr; QAction *m_costAbsolute = nullptr; QAction *m_costRelative = nullptr; QAction *m_costRelativeToParent = nullptr; QAction *m_cycleDetection = nullptr; QAction *m_shortenTemplates = nullptr; QComboBox *m_eventCombo = nullptr; QTimer m_updateTimer; QVector m_textMarks; QAction *m_startAction = nullptr; QAction *m_stopAction = nullptr; QAction *m_loadExternalLogFile = nullptr; QAction *m_startKCachegrind = nullptr; QAction *m_dumpAction = nullptr; QAction *m_resetAction = nullptr; QAction *m_pauseAction = nullptr; QAction *m_discardAction = nullptr; QString m_toggleCollectFunction; bool m_toolBusy = false; Perspective m_perspective{"Callgrind.Perspective", CallgrindTool::tr("Callgrind")}; }; CallgrindToolPrivate::CallgrindToolPrivate() { setObjectName("CallgrindTool"); m_updateTimer.setInterval(200); m_updateTimer.setSingleShot(true); m_proxyModel.setSourceModel(&m_dataModel); m_proxyModel.setDynamicSortFilter(true); m_proxyModel.setSortCaseSensitivity(Qt::CaseInsensitive); m_proxyModel.setFilterKeyColumn(DataModel::NameColumn); m_proxyModel.setFilterCaseSensitivity(Qt::CaseInsensitive); connect(&m_stackBrowser, &StackBrowser::currentChanged, this, &CallgrindToolPrivate::stackBrowserChanged); connect(&m_updateTimer, &QTimer::timeout, this, &CallgrindToolPrivate::updateFilterString); connect(EditorManager::instance(), &EditorManager::editorOpened, this, &CallgrindToolPrivate::editorOpened); m_startAction = Debugger::createStartAction(); m_stopAction = Debugger::createStopAction(); ActionContainer *menu = ActionManager::actionContainer(Debugger::Constants::M_DEBUG_ANALYZER); QString toolTip = CallgrindTool::tr("Valgrind Function Profiler uses the " "Callgrind tool to record function calls when a program runs."); if (!Utils::HostOsInfo::isWindowsHost()) { auto action = new QAction(CallgrindTool::tr("Valgrind Function Profiler"), this); action->setToolTip(toolTip); menu->addAction(ActionManager::registerAction(action, CallgrindLocalActionId), Debugger::Constants::G_ANALYZER_TOOLS); QObject::connect(action, &QAction::triggered, this, [this, action] { if (!Debugger::wantRunTool(OptimizedMode, action->text())) return; m_perspective.select(); ProjectExplorerPlugin::runStartupProject(CALLGRIND_RUN_MODE); }); QObject::connect(m_startAction, &QAction::triggered, action, &QAction::triggered); QObject::connect(m_startAction, &QAction::changed, action, [action, this] { action->setEnabled(m_startAction->isEnabled()); }); } auto action = new QAction(CallgrindTool::tr("Valgrind Function Profiler (External Application)"), this); action->setToolTip(toolTip); menu->addAction(ActionManager::registerAction(action, CallgrindRemoteActionId), Debugger::Constants::G_ANALYZER_REMOTE_TOOLS); QObject::connect(action, &QAction::triggered, this, [this, action] { auto runConfig = RunConfiguration::startupRunConfiguration(); if (!runConfig) { showCannotStartDialog(action->text()); return; } StartRemoteDialog dlg; if (dlg.exec() != QDialog::Accepted) return; m_perspective.select(); auto runControl = new RunControl(CALLGRIND_RUN_MODE); runControl->setRunConfiguration(runConfig); runControl->createMainWorker(); const auto runnable = dlg.runnable(); runControl->setRunnable(runnable); runControl->setDisplayName(runnable.executable.toUserOutput()); ProjectExplorerPlugin::startRunControl(runControl); }); // 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(Debugger::Constants::C_DEBUGMODE); editorContextMenu->addSeparator(analyzerContext); auto action = new QAction(CallgrindTool::tr("Profile Costs of This Function and Its Callees"), this); action->setIcon(ProjectExplorer::Icons::ANALYZER_START_SMALL.icon()); connect(action, &QAction::triggered, this, &CallgrindToolPrivate::handleShowCostsOfFunction); Command *cmd = ActionManager::registerAction(action, "Analyzer.Callgrind.ShowCostsOfFunction", analyzerContext); editorContextMenu->addAction(cmd); cmd->setAttribute(Command::CA_Hide); cmd->setAttribute(Command::CA_NonConfigurable); } QSettings *coreSettings = ICore::settings(); // // DockWidgets // m_visualization = new Visualization; m_visualization->setFrameStyle(QFrame::NoFrame); m_visualization->setObjectName("Valgrind.CallgrindTool.Visualisation"); m_visualization->setWindowTitle(CallgrindTool::tr("Visualization")); m_visualization->setModel(&m_dataModel); connect(m_visualization, &Visualization::functionActivated, this, &CallgrindToolPrivate::visualisationFunctionSelected); m_callersView = new CostView; m_callersView->setObjectName("Valgrind.CallgrindTool.CallersView"); m_callersView->setWindowTitle(CallgrindTool::tr("Callers")); m_callersView->setSettings(coreSettings, "Valgrind.CallgrindTool.CallersView"); m_callersView->sortByColumn(CallModel::CostColumn, Qt::DescendingOrder); m_callersView->setFrameStyle(QFrame::NoFrame); // enable sorting m_callersProxy.setSourceModel(&m_callersModel); m_callersView->setModel(&m_callersProxy); m_callersView->hideColumn(CallModel::CalleeColumn); connect(m_callersView, &QAbstractItemView::activated, this, &CallgrindToolPrivate::callerFunctionSelected); m_calleesView = new CostView; m_calleesView->setObjectName("Valgrind.CallgrindTool.CalleesView"); m_calleesView->setWindowTitle(CallgrindTool::tr("Callees")); m_calleesView->setSettings(coreSettings, "Valgrind.CallgrindTool.CalleesView"); m_calleesView->sortByColumn(CallModel::CostColumn, Qt::DescendingOrder); m_calleesView->setFrameStyle(QFrame::NoFrame); // enable sorting m_calleesProxy.setSourceModel(&m_calleesModel); m_calleesView->setModel(&m_calleesProxy); m_calleesView->hideColumn(CallModel::CallerColumn); connect(m_calleesView, &QAbstractItemView::activated, this, &CallgrindToolPrivate::calleeFunctionSelected); m_flatView = new CostView; m_flatView->setObjectName("Valgrind.CallgrindTool.FlatView"); m_flatView->setWindowTitle(CallgrindTool::tr("Functions")); m_flatView->setSettings(coreSettings, "Valgrind.CallgrindTool.FlatView"); m_flatView->sortByColumn(DataModel::SelfCostColumn, Qt::DescendingOrder); m_flatView->setFrameStyle(QFrame::NoFrame); m_flatView->setAttribute(Qt::WA_MacShowFocusRect, false); m_flatView->setModel(&m_proxyModel); connect(m_flatView, &QAbstractItemView::activated, this, &CallgrindToolPrivate::dataFunctionSelected); updateCostFormat(); ValgrindGlobalSettings *settings = ValgrindGlobalSettings::instance(); // // Control Widget // // load external log file action = m_loadExternalLogFile = new QAction(this); action->setIcon(Utils::Icons::OPENFILE_TOOLBAR.icon()); action->setToolTip(CallgrindTool::tr("Load External Log File")); connect(action, &QAction::triggered, this, &CallgrindToolPrivate::loadExternalLogFile); action = m_startKCachegrind = new QAction(this); action->setEnabled(false); const Utils::Icon kCachegrindIcon({{":/valgrind/images/kcachegrind.png", Theme::IconsBaseColor}}); action->setIcon(kCachegrindIcon.icon()); action->setToolTip(CallgrindTool::tr("Open results in KCachegrind.")); connect(action, &QAction::triggered, this, [this, settings] { QProcess::startDetached(settings->kcachegrindExecutable(), { m_lastFileName }); }); // dump action m_dumpAction = action = new QAction(this); action->setDisabled(true); action->setIcon(Utils::Icons::REDO.icon()); //action->setText(CallgrindTool::tr("Dump")); action->setToolTip(CallgrindTool::tr("Request the dumping of profile information. This will update the Callgrind visualization.")); connect(action, &QAction::triggered, this, &CallgrindToolPrivate::slotRequestDump); // reset action m_resetAction = action = new QAction(this); action->setDisabled(true); action->setIcon(Utils::Icons::RELOAD.icon()); //action->setText(CallgrindTool::tr("Reset")); action->setToolTip(CallgrindTool::tr("Reset all event counters.")); connect(action, &QAction::triggered, this, &CallgrindToolPrivate::resetRequested); // pause action m_pauseAction = action = new QAction(this); action->setCheckable(true); action->setIcon(Utils::Icons::INTERRUPT_SMALL_TOOLBAR.icon()); //action->setText(CallgrindTool::tr("Ignore")); action->setToolTip(CallgrindTool::tr("Pause event logging. No events are counted which will speed up program execution during profiling.")); connect(action, &QAction::toggled, this, &CallgrindToolPrivate::pauseToggled); // discard data action m_discardAction = action = new QAction(this); action->setIcon(Utils::Icons::CLEAN_TOOLBAR.icon()); action->setToolTip(CallgrindTool::tr("Discard Data")); connect(action, &QAction::triggered, this, [this](bool) { clearTextMarks(); doClear(true); }); // navigation // go back m_goBack = action = new QAction(this); action->setDisabled(true); action->setIcon(Utils::Icons::PREV_TOOLBAR.icon()); action->setToolTip(CallgrindTool::tr("Go back one step in history. This will select the previously selected item.")); connect(action, &QAction::triggered, &m_stackBrowser, &StackBrowser::goBack); // go forward m_goNext = action = new QAction(this); action->setDisabled(true); action->setIcon(Utils::Icons::NEXT_TOOLBAR.icon()); action->setToolTip(CallgrindTool::tr("Go forward one step in history.")); connect(action, &QAction::triggered, &m_stackBrowser, &StackBrowser::goNext); // event selection m_eventCombo = new QComboBox; m_eventCombo->setToolTip(CallgrindTool::tr("Selects which events from the profiling data are shown and visualized.")); connect(m_eventCombo, QOverload::of(&QComboBox::currentIndexChanged), this, &CallgrindToolPrivate::setCostEvent); updateEventCombo(); m_perspective.addToolBarAction(m_startAction); m_perspective.addToolBarAction(m_stopAction); m_perspective.addToolBarAction(m_loadExternalLogFile); m_perspective.addToolBarAction(m_startKCachegrind); m_perspective.addToolBarAction(m_dumpAction); m_perspective.addToolBarAction(m_resetAction); m_perspective.addToolBarAction(m_pauseAction); m_perspective.addToolBarAction(m_discardAction); m_perspective.addToolBarAction(m_goBack); m_perspective.addToolBarAction(m_goNext); m_perspective.addToolbarSeparator(); m_perspective.addToolBarWidget(m_eventCombo); // Cost formatting { auto group = new QActionGroup(this); // Show costs as absolute numbers m_costAbsolute = new QAction(CallgrindTool::tr("Absolute Costs"), this); m_costAbsolute->setToolTip(CallgrindTool::tr("Show costs as absolute numbers.")); m_costAbsolute->setCheckable(true); m_costAbsolute->setChecked(true); connect(m_costAbsolute, &QAction::toggled, this, &CallgrindToolPrivate::updateCostFormat); group->addAction(m_costAbsolute); // Show costs in percentages m_costRelative = new QAction(CallgrindTool::tr("Relative Costs"), this); m_costRelative->setToolTip(CallgrindTool::tr("Show costs relative to total inclusive cost.")); m_costRelative->setCheckable(true); connect(m_costRelative, &QAction::toggled, this, &CallgrindToolPrivate::updateCostFormat); group->addAction(m_costRelative); // Show costs relative to parent m_costRelativeToParent = new QAction(CallgrindTool::tr("Relative Costs to Parent"), this); m_costRelativeToParent->setToolTip(CallgrindTool::tr("Show costs relative to parent function's inclusive cost.")); m_costRelativeToParent->setCheckable(true); connect(m_costRelativeToParent, &QAction::toggled, this, &CallgrindToolPrivate::updateCostFormat); group->addAction(m_costRelativeToParent); auto button = new QToolButton; button->addActions(group->actions()); button->setPopupMode(QToolButton::InstantPopup); button->setText("$"); button->setToolTip(CallgrindTool::tr("Cost Format")); m_perspective.addToolBarWidget(button); } // Cycle detection //action = new QAction("Cycle Detection", this); ///FIXME: icon action = m_cycleDetection = new QAction("O", this); ///FIXME: icon action->setToolTip(CallgrindTool::tr("Enable cycle detection to properly handle recursive or circular function calls.")); action->setCheckable(true); connect(action, &QAction::toggled, &m_dataModel, &DataModel::enableCycleDetection); connect(action, &QAction::toggled, settings, &ValgrindGlobalSettings::setDetectCycles); // Shorter template signature action = m_shortenTemplates = new QAction("<>", this); action->setToolTip(CallgrindTool::tr("Remove template parameter lists when displaying function names.")); action->setCheckable(true); connect(action, &QAction::toggled, &m_dataModel, &DataModel::setShortenTemplates); connect(action, &QAction::toggled, settings, &ValgrindGlobalSettings::setShortenTemplates); // Filtering action = m_filterProjectCosts = new QAction(CallgrindTool::tr("Show Project Costs Only"), this); action->setIcon(Utils::Icons::FILTER.icon()); action->setToolTip(CallgrindTool::tr("Show only profiling info that originated from this project source.")); action->setCheckable(true); connect(action, &QAction::toggled, this, &CallgrindToolPrivate::handleFilterProjectCosts); // Filter ///FIXME: find workaround for https://bugreports.qt.io/browse/QTCREATORBUG-3247 m_searchFilter = new QLineEdit; m_searchFilter->setPlaceholderText(CallgrindTool::tr("Filter...")); connect(m_searchFilter, &QLineEdit::textChanged, &m_updateTimer, QOverload<>::of(&QTimer::start)); setCostFormat(settings->costFormat()); enableCycleDetection(settings->detectCycles()); m_perspective.addToolBarAction(m_cycleDetection); m_perspective.addToolBarAction(m_shortenTemplates); m_perspective.addToolBarAction(m_filterProjectCosts); m_perspective.addToolBarWidget(m_searchFilter); m_perspective.addWindow(m_flatView, Perspective::SplitVertical, nullptr); m_perspective.addWindow(m_calleesView, Perspective::SplitVertical, nullptr); m_perspective.addWindow(m_callersView, Perspective::SplitHorizontal, m_calleesView); m_perspective.addWindow(m_visualization, Perspective::SplitVertical, nullptr, false, Qt::RightDockWidgetArea); connect(ProjectExplorerPlugin::instance(), &ProjectExplorerPlugin::updateRunActions, this, &CallgrindToolPrivate::updateRunActions); } CallgrindToolPrivate::~CallgrindToolPrivate() { qDeleteAll(m_textMarks); delete m_flatView; delete m_callersView; delete m_calleesView; delete m_visualization; } void CallgrindToolPrivate::doClear(bool clearParseData) { if (clearParseData) // Crashed when done from destructor. setParseData(nullptr); // clear filters if (m_filterProjectCosts) m_filterProjectCosts->setChecked(false); m_proxyModel.setFilterBaseDir(QString()); if (m_searchFilter) m_searchFilter->clear(); m_proxyModel.setFilterFixedString(QString()); } void CallgrindToolPrivate::setBusyCursor(bool busy) { QCursor cursor(busy ? Qt::BusyCursor : Qt::ArrowCursor); m_flatView->setCursor(cursor); m_calleesView->setCursor(cursor); m_callersView->setCursor(cursor); m_visualization->setCursor(cursor); } void CallgrindToolPrivate::selectFunction(const Function *func) { if (!func) { m_flatView->clearSelection(); m_visualization->setFunction(nullptr); m_callersModel.clear(); m_calleesModel.clear(); return; } const QModelIndex index = m_dataModel.indexForObject(func); const QModelIndex proxyIndex = m_proxyModel.mapFromSource(index); m_flatView->selectionModel()->clearSelection(); m_flatView->selectionModel()->setCurrentIndex(proxyIndex, QItemSelectionModel::ClearAndSelect | QItemSelectionModel::Rows); m_flatView->scrollTo(proxyIndex); m_callersModel.setCalls(func->incomingCalls(), func); m_calleesModel.setCalls(func->outgoingCalls(), func); m_visualization->setFunction(func); const Function *item = m_stackBrowser.current(); if (!item || item != func) m_stackBrowser.select(func); if (QFile::exists(func->file())) { ///TODO: custom position support? int line = func->lineNumber(); EditorManager::openEditorAt(func->file(), qMax(line, 0)); } } void CallgrindToolPrivate::stackBrowserChanged() { m_goBack->setEnabled(m_stackBrowser.hasPrevious()); m_goNext->setEnabled(m_stackBrowser.hasNext()); const Function *item = m_stackBrowser.current(); selectFunction(item); } void CallgrindToolPrivate::updateFilterString() { m_proxyModel.setFilterFixedString(m_searchFilter->text()); } void CallgrindToolPrivate::setCostFormat(CostDelegate::CostFormat format) { switch (format) { case CostDelegate::FormatAbsolute: m_costAbsolute->setChecked(true); break; case CostDelegate::FormatRelative: m_costRelative->setChecked(true); break; case CostDelegate::FormatRelativeToParent: m_costRelativeToParent->setChecked(true); break; } } void CallgrindToolPrivate::setCostEvent(int index) { // prevent assert in model, don't try to set event to -1 // happens when we clear the eventcombo if (index == -1) index = 0; m_dataModel.setCostEvent(index); m_calleesModel.setCostEvent(index); m_callersModel.setCostEvent(index); } void CallgrindToolPrivate::enableCycleDetection(bool enabled) { m_cycleDetection->setChecked(enabled); } void CallgrindToolPrivate::shortenTemplates(bool enabled) { m_shortenTemplates->setChecked(enabled); } // Following functions can be called with actions=0 or widgets=0 // depending on initialization sequence (whether callgrind was current). CostDelegate::CostFormat CallgrindToolPrivate::costFormat() const { if (m_costRelativeToParent && m_costRelativeToParent->isChecked()) return CostDelegate::FormatRelativeToParent; if (m_costRelative && m_costRelative->isChecked()) return CostDelegate::FormatRelative; return CostDelegate::FormatAbsolute; } void CallgrindToolPrivate::updateCostFormat() { const CostDelegate::CostFormat format = costFormat(); if (m_flatView) m_flatView->setCostFormat(format); if (m_calleesView) { m_calleesView->setCostFormat(format); m_callersView->setCostFormat(format); } if (ValgrindGlobalSettings *settings = ValgrindGlobalSettings::instance()) settings->setCostFormat(format); } void CallgrindToolPrivate::handleFilterProjectCosts() { Project *pro = ProjectTree::currentProject(); if (pro && m_filterProjectCosts->isChecked()) { const QString projectDir = pro->projectDirectory().toString(); m_proxyModel.setFilterBaseDir(projectDir); } else { m_proxyModel.setFilterBaseDir(QString()); } } void CallgrindToolPrivate::dataFunctionSelected(const QModelIndex &index) { auto func = index.data(DataModel::FunctionRole).value(); QTC_ASSERT(func, return); selectFunction(func); } void CallgrindToolPrivate::calleeFunctionSelected(const QModelIndex &index) { auto call = index.data(CallModel::FunctionCallRole).value(); QTC_ASSERT(call, return); selectFunction(call->callee()); } void CallgrindToolPrivate::callerFunctionSelected(const QModelIndex &index) { auto call = index.data(CallModel::FunctionCallRole).value(); QTC_ASSERT(call, return); selectFunction(call->caller()); } void CallgrindToolPrivate::visualisationFunctionSelected(const Function *function) { if (function && function == m_visualization->function()) // up-navigation when the initial function was activated m_stackBrowser.goBack(); else selectFunction(function); } void CallgrindToolPrivate::setParseData(ParseData *data) { // we have new parse data, invalidate filters in the proxy model m_visualization->setFunction(nullptr); // invalidate parse data in the data model delete m_dataModel.parseData(); if (data && data->events().isEmpty()) { // might happen if the user cancelled the profile run // callgrind then sometimes produces empty callgrind.out.PID files delete data; data = nullptr; } m_lastFileName = data ? data->fileName() : QString(); m_dataModel.setParseData(data); m_calleesModel.setParseData(data); m_callersModel.setParseData(data); updateEventCombo(); // clear history for new data m_stackBrowser.clear(); // unset busy state //setBusy(false); } void CallgrindToolPrivate::updateEventCombo() { QTC_ASSERT(m_eventCombo, return); m_eventCombo->clear(); const ParseData *data = m_dataModel.parseData(); if (!data || data->events().isEmpty()) { m_eventCombo->hide(); return; } m_eventCombo->show(); foreach (const QString &event, data->events()) m_eventCombo->addItem(ParseData::prettyStringForEvent(event)); } void CallgrindToolPrivate::setupRunner(CallgrindToolRunner *toolRunner) { RunControl *runControl = toolRunner->runControl(); connect(toolRunner, &CallgrindToolRunner::parserDataReady, this, &CallgrindToolPrivate::takeParserDataFromRunControl); connect(runControl, &RunControl::stopped, this, &CallgrindToolPrivate::engineFinished); connect(this, &CallgrindToolPrivate::dumpRequested, toolRunner, &CallgrindToolRunner::dump); connect(this, &CallgrindToolPrivate::resetRequested, toolRunner, &CallgrindToolRunner::reset); connect(this, &CallgrindToolPrivate::pauseToggled, toolRunner, &CallgrindToolRunner::setPaused); connect(m_stopAction, &QAction::triggered, toolRunner, [runControl] { runControl->initiateStop(); }); // initialize run control toolRunner->setPaused(m_pauseAction->isChecked()); // we may want to toggle collect for one function only in this run toolRunner->setToggleCollectFunction(m_toggleCollectFunction); m_toggleCollectFunction.clear(); QTC_ASSERT(m_visualization, return); // apply project settings auto settings = qobject_cast(runControl->settings(ANALYZER_VALGRIND_SETTINGS)); if (settings) { m_visualization->setMinimumInclusiveCostRatio(settings->visualisationMinimumInclusiveCostRatio() / 100.0); m_proxyModel.setMinimumInclusiveCostRatio(settings->minimumInclusiveCostRatio() / 100.0); m_dataModel.setVerboseToolTipsEnabled(settings->enableEventToolTips()); } m_toolBusy = true; updateRunActions(); // enable/disable actions m_resetAction->setEnabled(true); m_dumpAction->setEnabled(true); m_loadExternalLogFile->setEnabled(false); clearTextMarks(); doClear(true); } void CallgrindToolPrivate::updateRunActions() { if (m_toolBusy) { m_startAction->setEnabled(false); m_startKCachegrind->setEnabled(false); m_startAction->setToolTip(CallgrindTool::tr("A Valgrind Callgrind analysis is still in progress.")); m_stopAction->setEnabled(true); } else { QString whyNot = CallgrindTool::tr("Start a Valgrind Callgrind analysis."); bool canRun = ProjectExplorerPlugin::canRunStartupProject(CALLGRIND_RUN_MODE, &whyNot); m_startAction->setToolTip(whyNot); m_startAction->setEnabled(canRun); m_stopAction->setEnabled(false); } } void CallgrindToolPrivate::clearTextMarks() { qDeleteAll(m_textMarks); m_textMarks.clear(); } void CallgrindToolPrivate::engineFinished() { m_toolBusy = false; updateRunActions(); // Enable/disable actions m_resetAction->setEnabled(false); m_dumpAction->setEnabled(false); m_loadExternalLogFile->setEnabled(true); const ParseData *data = m_dataModel.parseData(); if (data) showParserResults(data); else Debugger::showPermanentStatusMessage(CallgrindTool::tr("Profiling aborted.")); setBusyCursor(false); } void CallgrindToolPrivate::showParserResults(const ParseData *data) { QString msg; if (data) { // be careful, the list of events might be empty if (data->events().isEmpty()) { msg = CallgrindTool::tr("Parsing finished, no data."); } else { const QString costStr = QString::fromLatin1("%1 %2") .arg(QString::number(data->totalCost(0)), data->events().constFirst()); msg = CallgrindTool::tr("Parsing finished, total cost of %1 reported.").arg(costStr); } } else { msg = CallgrindTool::tr("Parsing failed."); } Debugger::showPermanentStatusMessage(msg); } void CallgrindToolPrivate::editorOpened(IEditor *editor) { if (auto widget = qobject_cast(editor->widget())) { connect(widget, &TextEditorWidget::markContextMenuRequested, this, &CallgrindToolPrivate::requestContextMenu); } } void CallgrindToolPrivate::requestContextMenu(TextEditorWidget *widget, int line, QMenu *menu) { // Find callgrind text mark that corresponds to this editor's file and line number foreach (CallgrindTextMark *textMark, m_textMarks) { if (textMark->fileName() == widget->textDocument()->filePath() && textMark->lineNumber() == line) { const Function *func = textMark->function(); QAction *action = menu->addAction(CallgrindTool::tr("Select This Function in the Analyzer Output")); connect(action, &QAction::triggered, this, [this, func] { selectFunction(func); }); break; } } } void CallgrindToolPrivate::handleShowCostsOfFunction() { CPlusPlus::Symbol *symbol = AnalyzerUtils::findSymbolUnderCursor(); if (!symbol) return; if (!symbol->isFunction()) return; CPlusPlus::Overview view; const QString qualifiedFunctionName = view.prettyName(CPlusPlus::LookupContext::fullyQualifiedName(symbol)); m_toggleCollectFunction = qualifiedFunctionName + "()"; m_startAction->trigger(); } void CallgrindToolPrivate::slotRequestDump() { //setBusy(true); m_visualization->setText(CallgrindTool::tr("Populating...")); emit dumpRequested(); } void CallgrindToolPrivate::loadExternalLogFile() { const QString filePath = QFileDialog::getOpenFileName( ICore::mainWindow(), CallgrindTool::tr("Open Callgrind Log File"), QString(), CallgrindTool::tr("Callgrind Output (callgrind.out*);;All Files (*)")); if (filePath.isEmpty()) return; QFile logFile(filePath); if (!logFile.open(QIODevice::ReadOnly | QIODevice::Text)) { QString msg = CallgrindTool::tr("Callgrind: Failed to open file for reading: %1").arg(filePath); TaskHub::addTask(Task::Error, msg, Debugger::Constants::ANALYZERTASK_ID); TaskHub::requestPopup(); return; } Debugger::showPermanentStatusMessage(CallgrindTool::tr("Parsing Profile Data...")); QCoreApplication::processEvents(); Parser parser; parser.parse(&logFile); takeParserData(parser.takeData()); } void CallgrindToolPrivate::takeParserDataFromRunControl(CallgrindToolRunner *rc) { takeParserData(rc->takeParserData()); } void CallgrindToolPrivate::takeParserData(ParseData *data) { showParserResults(data); if (!data) return; // clear first clearTextMarks(); doClear(true); setParseData(data); const QString kcachegrindExecutable = ValgrindGlobalSettings::instance()->kcachegrindExecutable(); const bool kcachegrindExists = !Utils::Environment::systemEnvironment().searchInPath( kcachegrindExecutable).isEmpty(); m_startKCachegrind->setEnabled(kcachegrindExists && !m_lastFileName.isEmpty()); createTextMarks(); } void CallgrindToolPrivate::createTextMarks() { QList locations; for (int row = 0; row < m_dataModel.rowCount(); ++row) { const QModelIndex index = m_dataModel.index(row, DataModel::InclusiveCostColumn); QString fileName = index.data(DataModel::FileNameRole).toString(); if (fileName.isEmpty() || fileName == "???") continue; bool ok = false; const int lineNumber = index.data(DataModel::LineNumberRole).toInt(&ok); QTC_ASSERT(ok, continue); // avoid creating invalid text marks if (lineNumber <= 0) continue; // sanitize filename, text marks need a canonical (i.e. no ".."s) path // BaseTextMark::editorOpened(Core::IEditor *editor) compares file names on string basis QFileInfo info(fileName); fileName = info.canonicalFilePath(); if (fileName.isEmpty()) continue; // isEmpty == true => file does not exist, continue then // create only one text mark per location const QString location = QString::fromLatin1("%1:%2").arg(fileName, QString::number(lineNumber)); if (locations.contains(location)) continue; locations << location; m_textMarks.append(new CallgrindTextMark(index, FilePath::fromString(fileName), lineNumber)); } } // CallgrindTool static CallgrindToolPrivate *dd = nullptr; void setupCallgrindRunner(CallgrindToolRunner *toolRunner) { dd->setupRunner(toolRunner); } CallgrindTool::CallgrindTool() { dd = new CallgrindToolPrivate; RunControl::registerWorker(CALLGRIND_RUN_MODE, {}); } CallgrindTool::~CallgrindTool() { delete dd; } } // namespace Internal } // namespace Valgrind #include "callgrindtool.moc"