blob: 0065309eca7c4170d240959b123bd8e0a059c4b9 [file] [log] [blame]
Zack Rusin601e8372011-03-24 22:23:21 -04001#include "mainwindow.h"
2
Zack Rusinf6667d12011-03-30 11:03:37 -04003#include "apitrace.h"
Zack Rusin96130ac2011-03-27 01:48:36 -04004#include "apitracecall.h"
Zack Rusin18eade52011-03-26 14:23:35 -04005#include "apicalldelegate.h"
Zack Rusin601e8372011-03-24 22:23:21 -04006#include "apitracemodel.h"
Zack Rusin91065372011-03-26 01:54:10 -04007#include "apitracefilter.h"
Zack Rusin3acde362011-04-06 01:11:55 -04008#include "retracer.h"
Zack Rusine1745572011-04-03 14:30:19 -04009#include "settingsdialog.h"
Zack Rusin3165c832011-04-07 20:32:46 -040010#include "ui_retracerdialog.h"
Zack Rusin8e7a4ff2011-04-07 01:15:48 -040011#include "vertexdatainterpreter.h"
Zack Rusin601e8372011-03-24 22:23:21 -040012
Zack Rusine8685f62011-03-30 20:14:01 -040013#include <qjson/parser.h>
14
Zack Rusin601e8372011-03-24 22:23:21 -040015#include <QAction>
16#include <QDebug>
Zack Rusinc97fadc2011-04-07 15:16:59 -040017#include <QDesktopServices>
Zack Rusin601e8372011-03-24 22:23:21 -040018#include <QDir>
19#include <QFileDialog>
Zack Rusin27cb2c42011-03-27 23:53:36 -040020#include <QLineEdit>
21#include <QMessageBox>
Zack Rusinde4ea412011-03-30 11:30:08 -040022#include <QProgressBar>
Zack Rusinea295452011-03-27 02:22:13 -040023#include <QToolBar>
Zack Rusinc97fadc2011-04-07 15:16:59 -040024#include <QUrl>
25#include <QWebPage>
Zack Rusin96130ac2011-03-27 01:48:36 -040026#include <QWebView>
Zack Rusin601e8372011-03-24 22:23:21 -040027
28
29MainWindow::MainWindow()
Zack Rusin27cb2c42011-03-27 23:53:36 -040030 : QMainWindow(),
Zack Rusin581e3ff2011-03-31 23:58:07 -040031 m_selectedEvent(0),
32 m_stateEvent(0),
Zack Rusin11f7e632011-03-30 22:47:51 -040033 m_jsonParser(new QJson::Parser())
Zack Rusin601e8372011-03-24 22:23:21 -040034{
35 m_ui.setupUi(this);
José Fonsecaf2c40fb2011-04-01 10:05:53 +010036 m_ui.stateTreeWidget->sortByColumn(0, Qt::AscendingOrder);
Zack Rusin601e8372011-03-24 22:23:21 -040037
Zack Rusinf6667d12011-03-30 11:03:37 -040038 m_trace = new ApiTrace();
Zack Rusinde4ea412011-03-30 11:30:08 -040039 connect(m_trace, SIGNAL(startedLoadingTrace()),
40 this, SLOT(startedLoadingTrace()));
41 connect(m_trace, SIGNAL(finishedLoadingTrace()),
42 this, SLOT(finishedLoadingTrace()));
Zack Rusinf6667d12011-03-30 11:03:37 -040043
Zack Rusin3acde362011-04-06 01:11:55 -040044 m_retracer = new Retracer(this);
45 connect(m_retracer, SIGNAL(finished(const QByteArray&)),
46 this, SLOT(replayFinished(const QByteArray&)));
47 connect(m_retracer, SIGNAL(error(const QString&)),
48 this, SLOT(replayError(const QString&)));
49
Zack Rusin8e7a4ff2011-04-07 01:15:48 -040050 m_vdataInterpreter = new VertexDataInterpreter(this);
51 m_vdataInterpreter->setListWidget(m_ui.vertexDataListWidget);
52 m_vdataInterpreter->setStride(
53 m_ui.vertexStrideSB->value());
54 m_vdataInterpreter->setComponents(
55 m_ui.vertexComponentsSB->value());
Zack Rusin1b917052011-04-07 19:38:40 -040056 m_vdataInterpreter->setStartingOffset(
57 m_ui.startingOffsetSB->value());
Zack Rusin8e7a4ff2011-04-07 01:15:48 -040058 m_vdataInterpreter->setTypeFromString(
59 m_ui.vertexTypeCB->currentText());
60
61 connect(m_ui.vertexInterpretButton, SIGNAL(clicked()),
62 m_vdataInterpreter, SLOT(interpretData()));
Zack Rusinb9341152011-04-07 19:09:28 -040063 connect(m_ui.vertexTypeCB, SIGNAL(currentIndexChanged(const QString&)),
Zack Rusin8e7a4ff2011-04-07 01:15:48 -040064 m_vdataInterpreter, SLOT(setTypeFromString(const QString&)));
65 connect(m_ui.vertexStrideSB, SIGNAL(valueChanged(int)),
66 m_vdataInterpreter, SLOT(setStride(int)));
67 connect(m_ui.vertexComponentsSB, SIGNAL(valueChanged(int)),
68 m_vdataInterpreter, SLOT(setComponents(int)));
Zack Rusin1b917052011-04-07 19:38:40 -040069 connect(m_ui.startingOffsetSB, SIGNAL(valueChanged(int)),
70 m_vdataInterpreter, SLOT(setStartingOffset(int)));
Zack Rusin8e7a4ff2011-04-07 01:15:48 -040071
Zack Rusin601e8372011-03-24 22:23:21 -040072 m_model = new ApiTraceModel();
Zack Rusinf6667d12011-03-30 11:03:37 -040073 m_model->setApiTrace(m_trace);
Zack Rusin91065372011-03-26 01:54:10 -040074 m_proxyModel = new ApiTraceFilter();
75 m_proxyModel->setSourceModel(m_model);
Zack Rusin96130ac2011-03-27 01:48:36 -040076 m_ui.callView->setModel(m_proxyModel);
Zack Rusin18eade52011-03-26 14:23:35 -040077 m_ui.callView->setItemDelegate(new ApiCallDelegate);
Zack Rusinc1acc7f2011-04-02 01:34:04 -040078 m_ui.callView->resizeColumnToContents(0);
79 m_ui.callView->header()->swapSections(0, 1);
80 m_ui.callView->setColumnWidth(1, 42);
Zack Rusin601e8372011-03-24 22:23:21 -040081
Zack Rusinea295452011-03-27 02:22:13 -040082 QToolBar *toolBar = addToolBar(tr("Navigation"));
83 m_filterEdit = new QLineEdit(toolBar);
84 toolBar->addWidget(m_filterEdit);
85
Zack Rusinde4ea412011-03-30 11:30:08 -040086 m_progressBar = new QProgressBar();
87 m_progressBar->setRange(0, 0);
88 statusBar()->addPermanentWidget(m_progressBar);
89 m_progressBar->hide();
90
Zack Rusin96130ac2011-03-27 01:48:36 -040091 m_ui.detailsDock->hide();
Zack Rusin8e7a4ff2011-04-07 01:15:48 -040092 m_ui.vertexDataDock->hide();
Zack Rusin11f7e632011-03-30 22:47:51 -040093 m_ui.stateDock->hide();
Zack Rusin8e7a4ff2011-04-07 01:15:48 -040094 setDockOptions(dockOptions() | QMainWindow::ForceTabbedDocks);
95
96 tabifyDockWidget(m_ui.stateDock, m_ui.vertexDataDock);
Zack Rusin96130ac2011-03-27 01:48:36 -040097
Zack Rusin601e8372011-03-24 22:23:21 -040098 connect(m_ui.actionOpen, SIGNAL(triggered()),
99 this, SLOT(openTrace()));
Zack Rusin27cb2c42011-03-27 23:53:36 -0400100 connect(m_ui.actionQuit, SIGNAL(triggered()),
101 this, SLOT(close()));
102
103 connect(m_ui.actionReplay, SIGNAL(triggered()),
104 this, SLOT(replayStart()));
105 connect(m_ui.actionStop, SIGNAL(triggered()),
106 this, SLOT(replayStop()));
Zack Rusin2caa06d2011-03-30 18:30:20 -0400107 connect(m_ui.actionLookupState, SIGNAL(triggered()),
108 this, SLOT(lookupState()));
Zack Rusine1745572011-04-03 14:30:19 -0400109 connect(m_ui.actionOptions, SIGNAL(triggered()),
110 this, SLOT(showSettings()));
Zack Rusin27cb2c42011-03-27 23:53:36 -0400111
Zack Rusin96130ac2011-03-27 01:48:36 -0400112 connect(m_ui.callView, SIGNAL(activated(const QModelIndex &)),
113 this, SLOT(callItemSelected(const QModelIndex &)));
Zack Rusinea295452011-03-27 02:22:13 -0400114 connect(m_filterEdit, SIGNAL(returnPressed()),
115 this, SLOT(filterTrace()));
Zack Rusinc97fadc2011-04-07 15:16:59 -0400116
117 m_ui.detailsWebView->page()->setLinkDelegationPolicy(
118 QWebPage::DelegateExternalLinks);
119 connect(m_ui.detailsWebView, SIGNAL(linkClicked(const QUrl&)),
120 this, SLOT(openHelp(const QUrl&)));
Zack Rusin601e8372011-03-24 22:23:21 -0400121}
122
123void MainWindow::openTrace()
124{
125 QString fileName =
126 QFileDialog::getOpenFileName(
127 this,
128 tr("Open Trace"),
129 QDir::homePath(),
130 tr("Trace Files (*.trace)"));
131
132 qDebug()<< "File name : " <<fileName;
133
Zack Rusin27cb2c42011-03-27 23:53:36 -0400134 newTraceFile(fileName);
Zack Rusin601e8372011-03-24 22:23:21 -0400135}
136
137void MainWindow::loadTrace(const QString &fileName)
138{
Zack Rusin27cb2c42011-03-27 23:53:36 -0400139 if (!QFile::exists(fileName)) {
140 QMessageBox::warning(this, tr("File Missing"),
141 tr("File '%1' doesn't exist.").arg(fileName));
142 return;
143 }
Zack Rusin601e8372011-03-24 22:23:21 -0400144 qDebug()<< "Loading : " <<fileName;
145
Zack Rusinde4ea412011-03-30 11:30:08 -0400146 m_progressBar->setValue(0);
Zack Rusin27cb2c42011-03-27 23:53:36 -0400147 newTraceFile(fileName);
Zack Rusin601e8372011-03-24 22:23:21 -0400148}
149
Zack Rusin96130ac2011-03-27 01:48:36 -0400150void MainWindow::callItemSelected(const QModelIndex &index)
151{
Zack Rusinc1acc7f2011-04-02 01:34:04 -0400152 ApiTraceEvent *event =
153 index.data(ApiTraceModel::EventRole).value<ApiTraceEvent*>();
154
155 if (event && event->type() == ApiTraceEvent::Call) {
156 ApiTraceCall *call = static_cast<ApiTraceCall*>(event);
Zack Rusin27cb2c42011-03-27 23:53:36 -0400157 m_ui.detailsWebView->setHtml(call->toHtml());
Zack Rusin96130ac2011-03-27 01:48:36 -0400158 m_ui.detailsDock->show();
Zack Rusin8e7a4ff2011-04-07 01:15:48 -0400159 if (call->hasBinaryData()) {
160 QByteArray data =
161 call->argValues[call->binaryDataIndex()].toByteArray();
162 m_vdataInterpreter->setData(data);
163
164 for (int i = 0; i < call->argNames.count(); ++i) {
165 QString name = call->argNames[i];
166 if (name == QLatin1String("stride")) {
167 int stride = call->argValues[i].toInt();
168 m_ui.vertexStrideSB->setValue(stride);
169 } else if (name == QLatin1String("size")) {
170 int components = call->argValues[i].toInt();
171 m_ui.vertexComponentsSB->setValue(components);
172 } else if (name == QLatin1String("type")) {
173 QString val = call->argValues[i].toString();
174 int textIndex = m_ui.vertexTypeCB->findText(val);
175 if (textIndex >= 0)
176 m_ui.vertexTypeCB->setCurrentIndex(textIndex);
177 }
178 }
179 }
180 m_ui.vertexDataDock->setVisible(call->hasBinaryData());
Zack Rusin581e3ff2011-03-31 23:58:07 -0400181 m_selectedEvent = call;
Zack Rusin96130ac2011-03-27 01:48:36 -0400182 } else {
Zack Rusinc1acc7f2011-04-02 01:34:04 -0400183 if (event && event->type() == ApiTraceEvent::Frame) {
184 m_selectedEvent = static_cast<ApiTraceFrame*>(event);
185 } else
186 m_selectedEvent = 0;
Zack Rusin96130ac2011-03-27 01:48:36 -0400187 m_ui.detailsDock->hide();
Zack Rusin8e7a4ff2011-04-07 01:15:48 -0400188 m_ui.vertexDataDock->hide();
Zack Rusin96130ac2011-03-27 01:48:36 -0400189 }
Zack Rusin581e3ff2011-03-31 23:58:07 -0400190 if (m_selectedEvent && !m_selectedEvent->state().isEmpty()) {
Zack Rusin11f7e632011-03-30 22:47:51 -0400191 fillStateForFrame();
192 } else
193 m_ui.stateDock->hide();
Zack Rusin96130ac2011-03-27 01:48:36 -0400194}
195
Zack Rusinea295452011-03-27 02:22:13 -0400196void MainWindow::filterTrace()
197{
198 m_proxyModel->setFilterString(m_filterEdit->text());
199}
200
Zack Rusin27cb2c42011-03-27 23:53:36 -0400201void MainWindow::replayStart()
202{
Zack Rusin3165c832011-04-07 20:32:46 -0400203 QDialog dlg;
204 Ui_RetracerDialog dlgUi;
205 dlgUi.setupUi(&dlg);
206
207 dlgUi.doubleBufferingCB->setChecked(
208 m_retracer->isDoubleBuffered());
209 dlgUi.benchmarkCB->setChecked(
210 m_retracer->isBenchmarking());
211
212 if (dlg.exec() == QDialog::Accepted) {
213 m_retracer->setDoubleBuffered(
214 dlgUi.doubleBufferingCB->isChecked());
215 m_retracer->setBenchmarking(
216 dlgUi.benchmarkCB->isChecked());
217 replayTrace(false);
218 }
Zack Rusin27cb2c42011-03-27 23:53:36 -0400219}
220
221void MainWindow::replayStop()
222{
Zack Rusin3acde362011-04-06 01:11:55 -0400223 m_retracer->terminate();
224 m_ui.actionStop->setEnabled(false);
225 m_ui.actionReplay->setEnabled(true);
226 m_ui.actionLookupState->setEnabled(true);
Zack Rusin27cb2c42011-03-27 23:53:36 -0400227}
228
229void MainWindow::newTraceFile(const QString &fileName)
230{
231 m_traceFileName = fileName;
Zack Rusinf6667d12011-03-30 11:03:37 -0400232 m_trace->setFileName(fileName);
Zack Rusin27cb2c42011-03-27 23:53:36 -0400233
234 if (m_traceFileName.isEmpty()) {
235 m_ui.actionReplay->setEnabled(false);
Zack Rusin2caa06d2011-03-30 18:30:20 -0400236 m_ui.actionLookupState->setEnabled(false);
Zack Rusina228d692011-04-06 17:29:39 -0400237 setWindowTitle(tr("QApiTrace"));
Zack Rusin27cb2c42011-03-27 23:53:36 -0400238 } else {
Zack Rusina228d692011-04-06 17:29:39 -0400239 QFileInfo info(fileName);
Zack Rusin27cb2c42011-03-27 23:53:36 -0400240 m_ui.actionReplay->setEnabled(true);
Zack Rusin2caa06d2011-03-30 18:30:20 -0400241 m_ui.actionLookupState->setEnabled(true);
Zack Rusina228d692011-04-06 17:29:39 -0400242 setWindowTitle(
243 tr("QApiTrace - %1").arg(info.fileName()));
Zack Rusin27cb2c42011-03-27 23:53:36 -0400244 }
245}
246
Zack Rusin3acde362011-04-06 01:11:55 -0400247void MainWindow::replayFinished(const QByteArray &output)
Zack Rusin27cb2c42011-03-27 23:53:36 -0400248{
249 m_ui.actionStop->setEnabled(false);
250 m_ui.actionReplay->setEnabled(true);
Zack Rusin2caa06d2011-03-30 18:30:20 -0400251 m_ui.actionLookupState->setEnabled(true);
Zack Rusin27cb2c42011-03-27 23:53:36 -0400252
Zack Rusin3acde362011-04-06 01:11:55 -0400253 if (m_retracer->captureState()) {
Zack Rusin11f7e632011-03-30 22:47:51 -0400254 bool ok = false;
255 QVariantMap parsedJson = m_jsonParser->parse(output, &ok).toMap();
256 parseState(parsedJson[QLatin1String("parameters")].toMap());
Zack Rusin2caa06d2011-03-30 18:30:20 -0400257 } else if (output.length() < 80) {
Zack Rusin27cb2c42011-03-27 23:53:36 -0400258 statusBar()->showMessage(output);
259 }
Zack Rusin581e3ff2011-03-31 23:58:07 -0400260 m_stateEvent = 0;
Zack Rusin27cb2c42011-03-27 23:53:36 -0400261}
262
Zack Rusin3acde362011-04-06 01:11:55 -0400263void MainWindow::replayError(const QString &message)
Zack Rusin27cb2c42011-03-27 23:53:36 -0400264{
265 m_ui.actionStop->setEnabled(false);
266 m_ui.actionReplay->setEnabled(true);
Zack Rusin2caa06d2011-03-30 18:30:20 -0400267 m_ui.actionLookupState->setEnabled(true);
Zack Rusin581e3ff2011-03-31 23:58:07 -0400268 m_stateEvent = 0;
Zack Rusin27cb2c42011-03-27 23:53:36 -0400269
Zack Rusin27cb2c42011-03-27 23:53:36 -0400270 QMessageBox::warning(
Zack Rusin3acde362011-04-06 01:11:55 -0400271 this, tr("Replay Failed"), message);
Zack Rusin27cb2c42011-03-27 23:53:36 -0400272}
273
Zack Rusinde4ea412011-03-30 11:30:08 -0400274void MainWindow::startedLoadingTrace()
275{
276 Q_ASSERT(m_trace);
277 m_progressBar->show();
278 QFileInfo info(m_trace->fileName());
279 statusBar()->showMessage(
280 tr("Loading %1...").arg(info.fileName()));
281}
282
283void MainWindow::finishedLoadingTrace()
284{
285 m_progressBar->hide();
286 if (!m_trace) {
287 return;
288 }
289 QFileInfo info(m_trace->fileName());
290 statusBar()->showMessage(
291 tr("Loaded %1").arg(info.fileName()), 3000);
292}
293
Zack Rusin2caa06d2011-03-30 18:30:20 -0400294void MainWindow::replayTrace(bool dumpState)
295{
Zack Rusin2caa06d2011-03-30 18:30:20 -0400296 if (m_traceFileName.isEmpty())
297 return;
298
Zack Rusin3acde362011-04-06 01:11:55 -0400299 m_retracer->setFileName(m_traceFileName);
300 m_retracer->setCaptureState(dumpState);
301 if (m_retracer->captureState() && m_selectedEvent) {
Zack Rusin581e3ff2011-03-31 23:58:07 -0400302 int index = 0;
303 if (m_selectedEvent->type() == ApiTraceEvent::Call) {
304 index = static_cast<ApiTraceCall*>(m_selectedEvent)->index;
305 } else if (m_selectedEvent->type() == ApiTraceEvent::Frame) {
306 ApiTraceFrame *frame = static_cast<ApiTraceFrame*>(m_selectedEvent);
307 if (frame->calls.isEmpty()) {
308 //XXX i guess we could still get the current state
309 qDebug()<<"tried to get a state for an empty frame";
310 return;
311 }
312 index = frame->calls.first()->index;
313 } else {
314 qDebug()<<"Unknown event type";
315 return;
316 }
Zack Rusin3acde362011-04-06 01:11:55 -0400317 m_retracer->setCaptureAtCallNumber(index);
Zack Rusin2caa06d2011-03-30 18:30:20 -0400318 }
Zack Rusin3acde362011-04-06 01:11:55 -0400319 m_retracer->start();
Zack Rusin2caa06d2011-03-30 18:30:20 -0400320
321 m_ui.actionStop->setEnabled(true);
322}
323
324void MainWindow::lookupState()
325{
Zack Rusin581e3ff2011-03-31 23:58:07 -0400326 if (!m_selectedEvent) {
Zack Rusin2caa06d2011-03-30 18:30:20 -0400327 QMessageBox::warning(
Zack Rusin581e3ff2011-03-31 23:58:07 -0400328 this, tr("Unknown Event"),
329 tr("To inspect the state select an event in the event list."));
Zack Rusin2caa06d2011-03-30 18:30:20 -0400330 return;
331 }
Zack Rusin581e3ff2011-03-31 23:58:07 -0400332 m_stateEvent = m_selectedEvent;
Zack Rusin2caa06d2011-03-30 18:30:20 -0400333 replayTrace(true);
334}
335
Zack Rusin11f7e632011-03-30 22:47:51 -0400336MainWindow::~MainWindow()
337{
338 delete m_jsonParser;
339}
340
341void MainWindow::parseState(const QVariantMap &params)
342{
343 QVariantMap::const_iterator itr;
344
Zack Rusin581e3ff2011-03-31 23:58:07 -0400345 m_stateEvent->setState(params);
Zack Rusinc1acc7f2011-04-02 01:34:04 -0400346 m_model->stateSetOnEvent(m_stateEvent);
Zack Rusin581e3ff2011-03-31 23:58:07 -0400347 if (m_selectedEvent == m_stateEvent) {
Zack Rusin11f7e632011-03-30 22:47:51 -0400348 fillStateForFrame();
349 } else {
350 m_ui.stateDock->hide();
351 }
352}
353
354static void
355variantToString(const QVariant &var, QString &str)
356{
357 if (var.type() == QVariant::List) {
358 QVariantList lst = var.toList();
359 str += QLatin1String("[");
360 for (int i = 0; i < lst.count(); ++i) {
361 QVariant val = lst[i];
362 variantToString(val, str);
363 if (i < lst.count() - 1)
364 str += QLatin1String(", ");
365 }
366 str += QLatin1String("]");
367 } else if (var.type() == QVariant::Map) {
368 Q_ASSERT(!"unsupported state type");
369 } else if (var.type() == QVariant::Hash) {
370 Q_ASSERT(!"unsupported state type");
371 } else {
372 str += var.toString();
373 }
374}
375
376void MainWindow::fillStateForFrame()
377{
378 QVariantMap::const_iterator itr;
379 QVariantMap params;
380
Zack Rusin581e3ff2011-03-31 23:58:07 -0400381 if (!m_selectedEvent || m_selectedEvent->state().isEmpty())
Zack Rusin11f7e632011-03-30 22:47:51 -0400382 return;
383
384 m_ui.stateTreeWidget->clear();
Zack Rusin581e3ff2011-03-31 23:58:07 -0400385 params = m_selectedEvent->state();
Zack Rusin11f7e632011-03-30 22:47:51 -0400386 QList<QTreeWidgetItem *> items;
387 for (itr = params.constBegin(); itr != params.constEnd(); ++itr) {
388 QString key = itr.key();
389 QString val;
390
391 variantToString(itr.value(), val);
392 //qDebug()<<"key = "<<key;
393 //qDebug()<<"val = "<<val;
394 QStringList lst;
395 lst += key;
396 lst += val;
397 items.append(new QTreeWidgetItem((QTreeWidget*)0, lst));
398 }
399 m_ui.stateTreeWidget->insertTopLevelItems(0, items);
400 m_ui.stateDock->show();
401}
402
Zack Rusine1745572011-04-03 14:30:19 -0400403void MainWindow::showSettings()
404{
405 SettingsDialog dialog;
406 dialog.setFilterOptions(m_proxyModel->filterOptions());
407
408 if (dialog.exec() == QDialog::Accepted) {
409 m_proxyModel->setFilterOptions(dialog.filterOptions());
410 }
411}
412
Zack Rusinc97fadc2011-04-07 15:16:59 -0400413void MainWindow::openHelp(const QUrl &url)
414{
415 QDesktopServices::openUrl(url);
416}
417
Zack Rusin601e8372011-03-24 22:23:21 -0400418#include "mainwindow.moc"