| 
									
										
										
										
											2011-04-04 14:39:28 +02:00
										 |  |  | /**************************************************************************
 | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** This file is part of Qt Creator | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** Copyright (c) 2011 Nokia Corporation and/or its subsidiary(-ies). | 
					
						
							|  |  |  | ** | 
					
						
							| 
									
										
										
										
											2011-11-02 15:59:12 +01:00
										 |  |  | ** Contact: Nokia Corporation (qt-info@nokia.com) | 
					
						
							| 
									
										
										
										
											2011-04-04 14:39:28 +02:00
										 |  |  | ** | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** GNU Lesser General Public License Usage | 
					
						
							|  |  |  | ** | 
					
						
							| 
									
										
										
										
											2011-07-06 15:48:52 +00:00
										 |  |  | ** This file may be used under the terms of the GNU Lesser General Public | 
					
						
							|  |  |  | ** License version 2.1 as published by the Free Software Foundation and | 
					
						
							|  |  |  | ** appearing in the file LICENSE.LGPL included in the packaging of this file. | 
					
						
							|  |  |  | ** Please review the following information to ensure the GNU Lesser General | 
					
						
							|  |  |  | ** Public License version 2.1 requirements will be met: | 
					
						
							|  |  |  | ** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
 | 
					
						
							| 
									
										
										
										
											2011-04-04 14:39:28 +02:00
										 |  |  | ** | 
					
						
							|  |  |  | ** In addition, as a special exception, Nokia gives you certain additional | 
					
						
							| 
									
										
										
										
											2011-07-06 15:48:52 +00:00
										 |  |  | ** rights. These rights are described in the Nokia Qt LGPL Exception | 
					
						
							| 
									
										
										
										
											2011-04-04 14:39:28 +02:00
										 |  |  | ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. | 
					
						
							|  |  |  | ** | 
					
						
							| 
									
										
										
										
											2011-07-06 15:48:52 +00:00
										 |  |  | ** Other Usage | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** Alternatively, this file may be used in accordance with the terms and | 
					
						
							|  |  |  | ** conditions contained in a signed written agreement between you and Nokia. | 
					
						
							|  |  |  | ** | 
					
						
							| 
									
										
										
										
											2011-04-04 14:39:28 +02:00
										 |  |  | ** If you have questions regarding the use of this file, please contact | 
					
						
							| 
									
										
										
										
											2011-11-02 15:59:12 +01:00
										 |  |  | ** Nokia at qt-info@nokia.com. | 
					
						
							| 
									
										
										
										
											2011-04-04 14:39:28 +02:00
										 |  |  | ** | 
					
						
							|  |  |  | **************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "modeltest.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <QtCore/QStringList>
 | 
					
						
							|  |  |  | #include <QtCore/QFile>
 | 
					
						
							|  |  |  | #include <QtCore/QTextStream>
 | 
					
						
							|  |  |  | #include <QtCore/QDebug>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <QtGui/QApplication>
 | 
					
						
							|  |  |  | #include <QtGui/QTableView>
 | 
					
						
							|  |  |  | #include <QtGui/QDesktopWidget>
 | 
					
						
							|  |  |  | #include <QtGui/QVBoxLayout>
 | 
					
						
							|  |  |  | #include <QtGui/QMenu>
 | 
					
						
							|  |  |  | #include <QtGui/QSortFilterProxyModel>
 | 
					
						
							|  |  |  | #include <QtGui/QStandardItemModel>
 | 
					
						
							|  |  |  | #include <QtGui/QPushButton>
 | 
					
						
							|  |  |  | #include <QtGui/QSpinBox>
 | 
					
						
							|  |  |  | #include <QtGui/QComboBox>
 | 
					
						
							|  |  |  | #include <QtGui/QMainWindow>
 | 
					
						
							|  |  |  | #include <QtGui/QDockWidget>
 | 
					
						
							|  |  |  | #include <QtGui/QToolButton>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-09 14:33:24 +02:00
										 |  |  | #include <callgrindcostview.h>
 | 
					
						
							|  |  |  | #include <callgrindcostdelegate.h>
 | 
					
						
							|  |  |  | #include <callgrindvisualisation.h>
 | 
					
						
							| 
									
										
										
										
											2011-04-04 14:39:28 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <valgrind/callgrind/callgrindparsedata.h>
 | 
					
						
							|  |  |  | #include <valgrind/callgrind/callgrindparser.h>
 | 
					
						
							|  |  |  | #include <valgrind/callgrind/callgrinddatamodel.h>
 | 
					
						
							|  |  |  | #include <valgrind/callgrind/callgrindfunction.h>
 | 
					
						
							|  |  |  | #include <valgrind/callgrind/callgrindcallmodel.h>
 | 
					
						
							|  |  |  | #include <valgrind/callgrind/callgrindcostitem.h>
 | 
					
						
							|  |  |  | #include <valgrind/callgrind/callgrindfunctioncall.h>
 | 
					
						
							|  |  |  | #include <valgrind/callgrind/callgrindproxymodel.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | using namespace Valgrind::Callgrind; | 
					
						
							|  |  |  | using namespace Callgrind::Internal; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QTextStream qerr(stderr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void usage() { | 
					
						
							|  |  |  |     qerr << "modeltest CALLGRINDFILE ..." << endl; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ModelTestWidget::ModelTestWidget(CallgrindWidgetHandler *handler) | 
					
						
							|  |  |  |     : m_format(0) | 
					
						
							|  |  |  |     , m_event(0) | 
					
						
							|  |  |  |     , m_handler(handler) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QVBoxLayout *l = new QVBoxLayout; | 
					
						
							|  |  |  |     setLayout(l); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QHBoxLayout *h = new QHBoxLayout; | 
					
						
							|  |  |  |     l->addLayout(h); | 
					
						
							|  |  |  |     l->addWidget(handler->flatView()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     m_handler->populateActions(h); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     m_format = new QComboBox; | 
					
						
							|  |  |  |     m_format->addItem("absolute", CostDelegate::FormatAbsolute); | 
					
						
							|  |  |  |     m_format->addItem("relative", CostDelegate::FormatRelative); | 
					
						
							|  |  |  |     m_format->addItem("rel. to parent", CostDelegate::FormatRelativeToParent); | 
					
						
							|  |  |  |     connect(m_format, SIGNAL(activated(int)), | 
					
						
							|  |  |  |             this, SLOT(formatChanged(int))); | 
					
						
							|  |  |  |     h->addWidget(m_format); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QDoubleSpinBox *minimumCost = new QDoubleSpinBox; | 
					
						
							|  |  |  |     minimumCost->setToolTip("Minimum cost"); | 
					
						
							|  |  |  |     minimumCost->setRange(0, 1); | 
					
						
							|  |  |  |     minimumCost->setDecimals(4); | 
					
						
							|  |  |  |     minimumCost->setSingleStep(0.01); | 
					
						
							|  |  |  |     connect(minimumCost, SIGNAL(valueChanged(double)), | 
					
						
							|  |  |  |             m_handler->proxyModel(), SLOT(setMinimumInclusiveCostRatio(double))); | 
					
						
							|  |  |  |     minimumCost->setValue(0.0001); | 
					
						
							|  |  |  |     h->addWidget(minimumCost); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     m_handler->flatView()->setContextMenuPolicy(Qt::CustomContextMenu); | 
					
						
							|  |  |  |     connect(m_handler->flatView(), SIGNAL(customContextMenuRequested(QPoint)), | 
					
						
							|  |  |  |             this, SLOT(showViewContextMenu(QPoint))); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     resize(qMin(qApp->desktop()->width(), 1024), 600); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ModelTestWidget::~ModelTestWidget() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ModelTestWidget::formatChanged(int index) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     CostDelegate::CostFormat format = static_cast<CostDelegate::CostFormat>(m_format->itemData(index).toInt()); | 
					
						
							|  |  |  |     m_handler->setCostFormat(format); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ModelTestWidget::showViewContextMenu(const QPoint &pos) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const QModelIndex idx = m_handler->flatView()->indexAt(pos); | 
					
						
							|  |  |  |     if (!idx.isValid()) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QMenu menu; | 
					
						
							|  |  |  |     QAction *showCostItems = menu.addAction("show cost items"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const ParseData* data = m_handler->dataModel()->parseData(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QAction *ret = menu.exec(m_handler->flatView()->mapToGlobal(pos)); | 
					
						
							|  |  |  |     if (ret == showCostItems) { | 
					
						
							|  |  |  |         ///TODO: put this into a reusable class?
 | 
					
						
							|  |  |  |         const Function *func = idx.data(DataModel::FunctionRole).value<const Function *>(); | 
					
						
							|  |  |  |         Q_ASSERT(func); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         QTableView *view = new QTableView(); | 
					
						
							|  |  |  |         view->setAttribute(Qt::WA_DeleteOnClose, true); | 
					
						
							|  |  |  |         const int rows = func->costItems().size(); | 
					
						
							|  |  |  |         const int columns = data->events().size() + data->positions().size() + 2; | 
					
						
							|  |  |  |         QStandardItemModel *model = new QStandardItemModel(rows, columns, view); | 
					
						
							|  |  |  |         int headerColumn = 0; | 
					
						
							|  |  |  |         foreach (const QString &event, data->events()) { | 
					
						
							|  |  |  |             model->setHeaderData(headerColumn++, Qt::Horizontal, event); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         const int lastEventColumn = headerColumn; | 
					
						
							|  |  |  |         foreach (const QString &pos, data->positions()) { | 
					
						
							|  |  |  |             model->setHeaderData(headerColumn++, Qt::Horizontal, pos); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         const int lastPosColumn = headerColumn; | 
					
						
							|  |  |  |         model->setHeaderData(headerColumn++, Qt::Horizontal, "Call"); | 
					
						
							|  |  |  |         model->setHeaderData(headerColumn++, Qt::Horizontal, "Differring File"); | 
					
						
							|  |  |  |         Q_ASSERT(headerColumn == columns); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         QVector<quint64> totalCosts; | 
					
						
							|  |  |  |         totalCosts.fill(0, data->events().size()); | 
					
						
							| 
									
										
										
										
											2011-05-11 16:26:34 +02:00
										 |  |  |         for (int row = 0; row < rows; ++row) { | 
					
						
							| 
									
										
										
										
											2011-04-04 14:39:28 +02:00
										 |  |  |             const CostItem *item = func->costItems().at(row); | 
					
						
							| 
									
										
										
										
											2011-05-11 16:26:34 +02:00
										 |  |  |             for (int column = 0; column < columns; ++column) { | 
					
						
							| 
									
										
										
										
											2011-04-04 14:39:28 +02:00
										 |  |  |                 QVariant value; | 
					
						
							|  |  |  |                 if (column < lastEventColumn) { | 
					
						
							|  |  |  |                     value = item->cost(column); | 
					
						
							|  |  |  |                     totalCosts[column] += item->cost(column); | 
					
						
							|  |  |  |                 } else if (column < lastPosColumn) { | 
					
						
							|  |  |  |                     value = item->position(column - lastEventColumn); | 
					
						
							|  |  |  |                 } else if (column == lastPosColumn) { | 
					
						
							|  |  |  |                     if (item->call()) | 
					
						
							|  |  |  |                         value = item->call()->callee()->name(); | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     value = item->differingFile(); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 model->setData(model->index(row, column), value); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         QStringList totalCostsStrings; | 
					
						
							| 
									
										
										
										
											2011-05-11 16:26:34 +02:00
										 |  |  |         for (int i = 0; i < totalCosts.size(); ++i) { | 
					
						
							| 
									
										
										
										
											2011-04-04 14:39:28 +02:00
										 |  |  |             totalCostsStrings << QString("%1: %2").arg(totalCosts.at(i)).arg(data->events().at(i)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         view->setWindowTitle(totalCostsStrings.join(QLatin1String(", "))); | 
					
						
							|  |  |  |         view->setModel(model); | 
					
						
							|  |  |  |         view->show(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int main(int argc, char *argv[]) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QApplication app(argc, argv); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (app.arguments().count() < 2) { | 
					
						
							|  |  |  |         usage(); | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ///TODO: multi-part callgrind files
 | 
					
						
							|  |  |  |     QFile file(app.arguments().at(1)); | 
					
						
							|  |  |  |     if (!file.open(QIODevice::ReadOnly)) { | 
					
						
							|  |  |  |         qerr << "could not open callgrind file for reading:" << file.fileName() << file.errorString(); | 
					
						
							|  |  |  |         return 3; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Parser p; | 
					
						
							|  |  |  |     p.parse(&file); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ParseData *data = p.takeData(); | 
					
						
							|  |  |  |     if (!data) { | 
					
						
							|  |  |  |         qerr << "invalid callgrind file:" << file.fileName() << endl; | 
					
						
							|  |  |  |         return 2; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QMainWindow window; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     CallgrindWidgetHandler *handler = new CallgrindWidgetHandler(&window); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ModelTestWidget *widget = new ModelTestWidget(handler); | 
					
						
							|  |  |  |     widget->setWindowTitle(file.fileName()); | 
					
						
							|  |  |  |     window.setCentralWidget(widget); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QDockWidget *callerDock = new QDockWidget("callers", &window); | 
					
						
							|  |  |  |     callerDock->setWidget(handler->callersView()); | 
					
						
							|  |  |  |     window.addDockWidget(Qt::RightDockWidgetArea, callerDock); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QDockWidget *calleeDock = new QDockWidget("callees", &window); | 
					
						
							|  |  |  |     calleeDock->setWidget(handler->calleesView()); | 
					
						
							|  |  |  |     window.addDockWidget(Qt::RightDockWidgetArea, calleeDock); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QDockWidget *widgetDock = new QDockWidget("visualisation", &window); | 
					
						
							|  |  |  |     widgetDock->setWidget(handler->visualisation()); | 
					
						
							|  |  |  |     window.addDockWidget(Qt::BottomDockWidgetArea, widgetDock); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     handler->setParseData(data); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     window.show(); | 
					
						
							|  |  |  |     return app.exec(); | 
					
						
							|  |  |  | } |