2013-08-08 13:28:08 +02:00
|
|
|
/****************************************************************************
|
|
|
|
**
|
2016-01-15 14:57:40 +01:00
|
|
|
** Copyright (C) 2016 The Qt Company Ltd.
|
|
|
|
** Contact: https://www.qt.io/licensing/
|
2013-08-08 13:28:08 +02:00
|
|
|
**
|
|
|
|
** 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
|
2016-01-15 14:57:40 +01:00
|
|
|
** 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.
|
2013-08-08 13:28:08 +02:00
|
|
|
**
|
2016-01-15 14:57:40 +01:00
|
|
|
** 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.
|
2013-08-08 13:28:08 +02:00
|
|
|
**
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#include "qmlprofilermodelmanager.h"
|
2015-02-10 20:29:54 +01:00
|
|
|
#include "qmlprofilerconstants.h"
|
2014-02-18 17:32:20 +01:00
|
|
|
#include "qmlprofilerdatamodel.h"
|
2013-08-08 13:28:08 +02:00
|
|
|
#include "qmlprofilertracefile.h"
|
2014-10-27 17:41:22 +01:00
|
|
|
#include "qmlprofilernotesmodel.h"
|
2013-08-08 13:28:08 +02:00
|
|
|
|
2015-02-10 20:29:54 +01:00
|
|
|
#include <coreplugin/progressmanager/progressmanager.h>
|
|
|
|
#include <utils/runextensions.h>
|
2013-08-08 13:28:08 +02:00
|
|
|
#include <utils/qtcassert.h>
|
|
|
|
|
|
|
|
#include <QDebug>
|
|
|
|
#include <QFile>
|
2014-11-03 14:19:26 +01:00
|
|
|
#include <QMessageBox>
|
2013-08-08 13:28:08 +02:00
|
|
|
|
2016-04-28 16:13:16 +02:00
|
|
|
#include <functional>
|
|
|
|
|
2013-08-08 13:28:08 +02:00
|
|
|
namespace QmlProfiler {
|
|
|
|
namespace Internal {
|
|
|
|
|
2015-11-13 17:55:58 +01:00
|
|
|
static const char *ProfileFeatureNames[] = {
|
2014-09-09 18:22:58 +02:00
|
|
|
QT_TRANSLATE_NOOP("MainView", "JavaScript"),
|
|
|
|
QT_TRANSLATE_NOOP("MainView", "Memory Usage"),
|
|
|
|
QT_TRANSLATE_NOOP("MainView", "Pixmap Cache"),
|
|
|
|
QT_TRANSLATE_NOOP("MainView", "Scene Graph"),
|
|
|
|
QT_TRANSLATE_NOOP("MainView", "Animations"),
|
|
|
|
QT_TRANSLATE_NOOP("MainView", "Painting"),
|
|
|
|
QT_TRANSLATE_NOOP("MainView", "Compiling"),
|
|
|
|
QT_TRANSLATE_NOOP("MainView", "Creating"),
|
|
|
|
QT_TRANSLATE_NOOP("MainView", "Binding"),
|
|
|
|
QT_TRANSLATE_NOOP("MainView", "Handling Signal"),
|
2015-11-13 17:55:58 +01:00
|
|
|
QT_TRANSLATE_NOOP("MainView", "Input Events"),
|
|
|
|
QT_TRANSLATE_NOOP("MainView", "Debug Messages")
|
2014-09-09 18:22:58 +02:00
|
|
|
};
|
2013-08-08 13:28:08 +02:00
|
|
|
|
2016-05-02 12:18:57 +02:00
|
|
|
Q_STATIC_ASSERT(sizeof(ProfileFeatureNames) == sizeof(char *) * MaximumProfileFeature);
|
2015-11-13 17:55:58 +01:00
|
|
|
|
2013-08-08 13:28:08 +02:00
|
|
|
/////////////////////////////////////////////////////////////////////
|
2014-06-03 15:35:54 +02:00
|
|
|
QmlProfilerTraceTime::QmlProfilerTraceTime(QObject *parent) :
|
2016-04-28 16:13:16 +02:00
|
|
|
QObject(parent), m_startTime(-1), m_endTime(-1),
|
|
|
|
m_restrictedStartTime(-1), m_restrictedEndTime(-1)
|
2013-08-08 13:28:08 +02:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
qint64 QmlProfilerTraceTime::startTime() const
|
|
|
|
{
|
2016-04-28 16:13:16 +02:00
|
|
|
return m_restrictedStartTime != -1 ? m_restrictedStartTime : m_startTime;
|
2013-08-08 13:28:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
qint64 QmlProfilerTraceTime::endTime() const
|
|
|
|
{
|
2016-04-28 16:13:16 +02:00
|
|
|
return m_restrictedEndTime != -1 ? m_restrictedEndTime : m_endTime;
|
2013-08-08 13:28:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
qint64 QmlProfilerTraceTime::duration() const
|
|
|
|
{
|
|
|
|
return endTime() - startTime();
|
|
|
|
}
|
|
|
|
|
2016-04-28 16:13:16 +02:00
|
|
|
bool QmlProfilerTraceTime::isRestrictedToRange() const
|
|
|
|
{
|
|
|
|
return m_restrictedStartTime != -1 || m_restrictedEndTime != -1;
|
|
|
|
}
|
|
|
|
|
2013-08-08 13:28:08 +02:00
|
|
|
void QmlProfilerTraceTime::clear()
|
|
|
|
{
|
2016-04-28 16:13:16 +02:00
|
|
|
restrictToRange(-1, -1);
|
2014-10-14 16:44:45 +02:00
|
|
|
setTime(-1, -1);
|
2013-08-08 13:28:08 +02:00
|
|
|
}
|
|
|
|
|
2014-10-14 16:44:45 +02:00
|
|
|
void QmlProfilerTraceTime::setTime(qint64 startTime, qint64 endTime)
|
2013-08-08 13:28:08 +02:00
|
|
|
{
|
2016-04-28 16:13:16 +02:00
|
|
|
QTC_ASSERT(startTime <= endTime, endTime = startTime);
|
|
|
|
m_startTime = startTime;
|
|
|
|
m_endTime = endTime;
|
2013-08-08 13:28:08 +02:00
|
|
|
}
|
|
|
|
|
2014-03-12 17:20:52 +01:00
|
|
|
void QmlProfilerTraceTime::decreaseStartTime(qint64 time)
|
|
|
|
{
|
2015-11-13 17:54:08 +01:00
|
|
|
if (m_startTime > time || m_startTime == -1) {
|
2014-10-14 16:44:45 +02:00
|
|
|
m_startTime = time;
|
2015-11-13 17:54:08 +01:00
|
|
|
if (m_endTime == -1)
|
|
|
|
m_endTime = m_startTime;
|
|
|
|
else
|
|
|
|
QTC_ASSERT(m_endTime >= m_startTime, m_endTime = m_startTime);
|
2014-10-14 16:44:45 +02:00
|
|
|
}
|
2014-03-12 17:20:52 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void QmlProfilerTraceTime::increaseEndTime(qint64 time)
|
|
|
|
{
|
2015-11-13 17:54:08 +01:00
|
|
|
if (m_endTime < time || m_endTime == -1) {
|
2014-10-14 16:44:45 +02:00
|
|
|
m_endTime = time;
|
2015-11-13 17:54:08 +01:00
|
|
|
if (m_startTime == -1)
|
|
|
|
m_startTime = m_endTime;
|
|
|
|
else
|
|
|
|
QTC_ASSERT(m_endTime >= m_startTime, m_startTime = m_endTime);
|
2014-10-14 16:44:45 +02:00
|
|
|
}
|
2014-03-12 17:20:52 +01:00
|
|
|
}
|
|
|
|
|
2016-04-28 16:13:16 +02:00
|
|
|
void QmlProfilerTraceTime::restrictToRange(qint64 startTime, qint64 endTime)
|
|
|
|
{
|
|
|
|
QTC_ASSERT(endTime == -1 || startTime <= endTime, endTime = startTime);
|
|
|
|
m_restrictedStartTime = startTime;
|
|
|
|
m_restrictedEndTime = endTime;
|
|
|
|
}
|
|
|
|
|
2013-08-08 13:28:08 +02:00
|
|
|
|
|
|
|
} // namespace Internal
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
class QmlProfilerModelManager::QmlProfilerModelManagerPrivate
|
|
|
|
{
|
|
|
|
public:
|
2014-02-18 17:32:20 +01:00
|
|
|
QmlProfilerDataModel *model;
|
2014-10-27 17:41:22 +01:00
|
|
|
QmlProfilerNotesModel *notesModel;
|
2014-09-26 15:49:49 +02:00
|
|
|
|
2015-09-10 13:51:54 +02:00
|
|
|
QmlProfilerModelManager::State state;
|
2013-08-08 13:28:08 +02:00
|
|
|
QmlProfilerTraceTime *traceTime;
|
|
|
|
|
2016-05-11 13:58:20 +02:00
|
|
|
int numRegisteredModels;
|
2016-05-11 14:43:26 +02:00
|
|
|
int numFinishedFinalizers;
|
|
|
|
|
|
|
|
uint numLoadedEvents;
|
2015-06-30 15:55:33 +02:00
|
|
|
quint64 availableFeatures;
|
|
|
|
quint64 visibleFeatures;
|
|
|
|
quint64 recordedFeatures;
|
2016-08-04 17:29:05 +02:00
|
|
|
bool aggregateTraces;
|
2016-04-28 16:19:17 +02:00
|
|
|
|
|
|
|
QHash<ProfileFeature, QVector<EventLoader> > eventLoaders;
|
|
|
|
QVector<Finalizer> finalizers;
|
2013-08-08 13:28:08 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
QmlProfilerModelManager::QmlProfilerModelManager(Utils::FileInProjectFinder *finder, QObject *parent) :
|
2016-04-28 16:02:54 +02:00
|
|
|
QObject(parent), d(new QmlProfilerModelManagerPrivate)
|
2013-08-08 13:28:08 +02:00
|
|
|
{
|
2016-05-11 13:58:20 +02:00
|
|
|
d->numRegisteredModels = 0;
|
2016-05-11 14:43:26 +02:00
|
|
|
d->numFinishedFinalizers = 0;
|
|
|
|
d->numLoadedEvents = 0;
|
2015-06-30 15:55:33 +02:00
|
|
|
d->availableFeatures = 0;
|
|
|
|
d->visibleFeatures = 0;
|
|
|
|
d->recordedFeatures = 0;
|
2016-08-04 17:29:05 +02:00
|
|
|
d->aggregateTraces = false;
|
2014-02-18 17:32:20 +01:00
|
|
|
d->model = new QmlProfilerDataModel(finder, this);
|
2015-09-10 13:51:54 +02:00
|
|
|
d->state = Empty;
|
2013-08-08 13:28:08 +02:00
|
|
|
d->traceTime = new QmlProfilerTraceTime(this);
|
2014-10-27 17:41:22 +01:00
|
|
|
d->notesModel = new QmlProfilerNotesModel(this);
|
2016-04-28 16:13:16 +02:00
|
|
|
connect(d->model, &QmlProfilerDataModel::allTypesLoaded,
|
|
|
|
this, &QmlProfilerModelManager::processingDone);
|
2013-08-08 13:28:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
QmlProfilerModelManager::~QmlProfilerModelManager()
|
|
|
|
{
|
|
|
|
delete d;
|
|
|
|
}
|
|
|
|
|
|
|
|
QmlProfilerTraceTime *QmlProfilerModelManager::traceTime() const
|
|
|
|
{
|
|
|
|
return d->traceTime;
|
|
|
|
}
|
|
|
|
|
2014-02-18 17:32:20 +01:00
|
|
|
QmlProfilerDataModel *QmlProfilerModelManager::qmlModel() const
|
2013-08-08 13:28:08 +02:00
|
|
|
{
|
|
|
|
return d->model;
|
|
|
|
}
|
|
|
|
|
2014-10-27 17:41:22 +01:00
|
|
|
QmlProfilerNotesModel *QmlProfilerModelManager::notesModel() const
|
2014-09-26 15:49:49 +02:00
|
|
|
{
|
|
|
|
return d->notesModel;
|
|
|
|
}
|
|
|
|
|
2013-08-08 13:28:08 +02:00
|
|
|
bool QmlProfilerModelManager::isEmpty() const
|
|
|
|
{
|
2015-05-26 14:41:27 +02:00
|
|
|
return d->model->isEmpty();
|
2013-08-08 13:28:08 +02:00
|
|
|
}
|
|
|
|
|
2016-05-11 14:43:26 +02:00
|
|
|
uint QmlProfilerModelManager::numLoadedEvents() const
|
|
|
|
{
|
|
|
|
return d->numLoadedEvents;
|
|
|
|
}
|
|
|
|
|
2013-08-08 13:28:08 +02:00
|
|
|
int QmlProfilerModelManager::registerModelProxy()
|
|
|
|
{
|
2016-05-11 13:58:20 +02:00
|
|
|
return d->numRegisteredModels++;
|
2013-08-08 13:28:08 +02:00
|
|
|
}
|
|
|
|
|
2016-05-11 14:43:26 +02:00
|
|
|
int QmlProfilerModelManager::numFinishedFinalizers() const
|
|
|
|
{
|
|
|
|
return d->numFinishedFinalizers;
|
|
|
|
}
|
|
|
|
|
|
|
|
int QmlProfilerModelManager::numRegisteredFinalizers() const
|
|
|
|
{
|
|
|
|
return d->finalizers.count();
|
|
|
|
}
|
|
|
|
|
2016-04-28 16:19:17 +02:00
|
|
|
void QmlProfilerModelManager::dispatch(const QmlEvent &event, const QmlEventType &type)
|
|
|
|
{
|
|
|
|
foreach (const EventLoader &loader, d->eventLoaders[type.feature()])
|
|
|
|
loader(event, type);
|
2016-05-11 14:43:26 +02:00
|
|
|
++d->numLoadedEvents;
|
2016-04-28 16:19:17 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void QmlProfilerModelManager::announceFeatures(quint64 features, EventLoader eventLoader,
|
|
|
|
Finalizer finalizer)
|
2014-09-09 18:22:58 +02:00
|
|
|
{
|
2015-06-30 15:55:33 +02:00
|
|
|
if ((features & d->availableFeatures) != features) {
|
|
|
|
d->availableFeatures |= features;
|
|
|
|
emit availableFeaturesChanged(d->availableFeatures);
|
2014-09-09 18:22:58 +02:00
|
|
|
}
|
2015-06-30 15:55:33 +02:00
|
|
|
if ((features & d->visibleFeatures) != features) {
|
|
|
|
d->visibleFeatures |= features;
|
|
|
|
emit visibleFeaturesChanged(d->visibleFeatures);
|
|
|
|
}
|
2016-04-28 16:19:17 +02:00
|
|
|
|
|
|
|
for (int feature = 0; feature != MaximumProfileFeature; ++feature) {
|
2016-05-26 09:27:15 +02:00
|
|
|
if (features & (1ULL << feature))
|
2016-04-28 16:19:17 +02:00
|
|
|
d->eventLoaders[static_cast<ProfileFeature>(feature)].append(eventLoader);
|
|
|
|
}
|
|
|
|
|
|
|
|
d->finalizers.append(finalizer);
|
2015-06-30 15:55:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
quint64 QmlProfilerModelManager::availableFeatures() const
|
|
|
|
{
|
|
|
|
return d->availableFeatures;
|
2014-09-09 18:22:58 +02:00
|
|
|
}
|
|
|
|
|
2015-06-30 15:55:33 +02:00
|
|
|
quint64 QmlProfilerModelManager::visibleFeatures() const
|
2014-09-09 18:22:58 +02:00
|
|
|
{
|
2015-06-30 15:55:33 +02:00
|
|
|
return d->visibleFeatures;
|
|
|
|
}
|
|
|
|
|
|
|
|
void QmlProfilerModelManager::setVisibleFeatures(quint64 features)
|
|
|
|
{
|
|
|
|
if (d->visibleFeatures != features) {
|
|
|
|
d->visibleFeatures = features;
|
|
|
|
emit visibleFeaturesChanged(d->visibleFeatures);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
quint64 QmlProfilerModelManager::recordedFeatures() const
|
|
|
|
{
|
|
|
|
return d->recordedFeatures;
|
|
|
|
}
|
|
|
|
|
|
|
|
void QmlProfilerModelManager::setRecordedFeatures(quint64 features)
|
|
|
|
{
|
|
|
|
if (d->recordedFeatures != features) {
|
|
|
|
d->recordedFeatures = features;
|
|
|
|
emit recordedFeaturesChanged(d->recordedFeatures);
|
|
|
|
}
|
2014-09-09 18:22:58 +02:00
|
|
|
}
|
|
|
|
|
2016-08-04 17:29:05 +02:00
|
|
|
bool QmlProfilerModelManager::aggregateTraces() const
|
|
|
|
{
|
|
|
|
return d->aggregateTraces;
|
|
|
|
}
|
|
|
|
|
|
|
|
void QmlProfilerModelManager::setAggregateTraces(bool aggregateTraces)
|
|
|
|
{
|
|
|
|
d->aggregateTraces = aggregateTraces;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-05-02 12:18:57 +02:00
|
|
|
const char *QmlProfilerModelManager::featureName(ProfileFeature feature)
|
2014-09-09 18:22:58 +02:00
|
|
|
{
|
|
|
|
return ProfileFeatureNames[feature];
|
|
|
|
}
|
|
|
|
|
2015-09-10 17:11:21 +02:00
|
|
|
void QmlProfilerModelManager::acquiringDone()
|
2013-08-08 13:28:08 +02:00
|
|
|
{
|
2015-09-10 17:11:21 +02:00
|
|
|
QTC_ASSERT(state() == AcquiringData, /**/);
|
|
|
|
setState(ProcessingData);
|
2016-04-28 16:13:16 +02:00
|
|
|
d->model->finalize();
|
2015-09-10 17:11:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void QmlProfilerModelManager::processingDone()
|
|
|
|
{
|
|
|
|
QTC_ASSERT(state() == ProcessingData, /**/);
|
2015-09-25 10:35:42 +02:00
|
|
|
// Load notes after the timeline models have been initialized ...
|
|
|
|
// which happens on stateChanged(Done).
|
2016-04-28 16:13:16 +02:00
|
|
|
|
2016-05-11 14:43:26 +02:00
|
|
|
foreach (const Finalizer &finalizer, d->finalizers) {
|
2016-04-28 16:13:16 +02:00
|
|
|
finalizer();
|
2016-05-11 14:43:26 +02:00
|
|
|
++d->numFinishedFinalizers;
|
|
|
|
}
|
2016-04-28 16:13:16 +02:00
|
|
|
|
2015-09-25 10:35:42 +02:00
|
|
|
d->notesModel->loadData();
|
2016-04-28 16:13:16 +02:00
|
|
|
setState(Done);
|
|
|
|
|
2015-09-10 17:11:21 +02:00
|
|
|
emit loadFinished();
|
2013-08-08 13:28:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void QmlProfilerModelManager::save(const QString &filename)
|
|
|
|
{
|
2015-02-10 20:29:54 +01:00
|
|
|
QFile *file = new QFile(filename);
|
|
|
|
if (!file->open(QIODevice::WriteOnly)) {
|
2013-08-08 13:28:08 +02:00
|
|
|
emit error(tr("Could not open %1 for writing.").arg(filename));
|
2015-02-10 20:29:54 +01:00
|
|
|
delete file;
|
|
|
|
emit saveFinished();
|
2013-08-08 13:28:08 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-07-06 11:34:15 +02:00
|
|
|
d->notesModel->saveData();
|
2015-02-10 20:29:54 +01:00
|
|
|
|
2016-02-08 16:51:48 +01:00
|
|
|
QmlProfilerFileWriter *writer = new QmlProfilerFileWriter(this);
|
|
|
|
writer->setTraceTime(traceTime()->startTime(), traceTime()->endTime(),
|
|
|
|
traceTime()->duration());
|
2016-04-29 14:43:32 +02:00
|
|
|
writer->setData(d->model);
|
2016-04-26 12:26:46 +02:00
|
|
|
writer->setNotes(d->notesModel->notes());
|
2016-02-08 16:51:48 +01:00
|
|
|
|
|
|
|
connect(writer, &QObject::destroyed, this, &QmlProfilerModelManager::saveFinished,
|
|
|
|
Qt::QueuedConnection);
|
|
|
|
|
2016-02-08 16:26:19 +01:00
|
|
|
QFuture<void> result = Utils::runAsync([file, writer] (QFutureInterface<void> &future) {
|
2016-02-08 16:51:48 +01:00
|
|
|
writer->setFuture(&future);
|
2016-05-10 13:27:48 +02:00
|
|
|
if (file->fileName().endsWith(QLatin1String(Constants::QtdFileExtension)))
|
|
|
|
writer->saveQtd(file);
|
|
|
|
else
|
|
|
|
writer->saveQzt(file);
|
2016-10-26 10:47:21 +02:00
|
|
|
writer->deleteLater();
|
2015-02-10 20:29:54 +01:00
|
|
|
file->deleteLater();
|
|
|
|
});
|
|
|
|
|
|
|
|
Core::ProgressManager::addTask(result, tr("Saving Trace Data"), Constants::TASK_SAVE,
|
|
|
|
Core::ProgressManager::ShowInApplicationIcon);
|
2013-08-08 13:28:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void QmlProfilerModelManager::load(const QString &filename)
|
|
|
|
{
|
2016-05-10 13:27:48 +02:00
|
|
|
bool isQtd = filename.endsWith(QLatin1String(Constants::QtdFileExtension));
|
2015-09-10 11:49:17 +02:00
|
|
|
QFile *file = new QFile(filename, this);
|
2016-05-10 13:27:48 +02:00
|
|
|
if (!file->open(isQtd ? (QIODevice::ReadOnly | QIODevice::Text) : QIODevice::ReadOnly)) {
|
2015-09-10 11:49:17 +02:00
|
|
|
emit error(tr("Could not open %1 for reading.").arg(filename));
|
2015-02-17 14:25:24 +01:00
|
|
|
delete file;
|
|
|
|
emit loadFinished();
|
2013-08-08 13:28:08 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
clear();
|
2015-09-10 13:51:54 +02:00
|
|
|
setState(AcquiringData);
|
2016-02-08 16:51:48 +01:00
|
|
|
QmlProfilerFileReader *reader = new QmlProfilerFileReader(this);
|
|
|
|
|
|
|
|
connect(reader, &QmlProfilerFileReader::error, this, [this, reader](const QString &message) {
|
|
|
|
delete reader;
|
|
|
|
emit error(message);
|
|
|
|
}, Qt::QueuedConnection);
|
|
|
|
|
2016-05-10 13:27:48 +02:00
|
|
|
connect(reader, &QmlProfilerFileReader::typesLoaded,
|
|
|
|
d->model, &QmlProfilerDataModel::setEventTypes);
|
|
|
|
|
|
|
|
connect(reader, &QmlProfilerFileReader::notesLoaded,
|
|
|
|
d->notesModel, &QmlProfilerNotesModel::setNotes);
|
|
|
|
|
2016-11-08 17:26:39 +01:00
|
|
|
connect(reader, &QmlProfilerFileReader::qmlEventsLoaded,
|
|
|
|
d->model, &QmlProfilerDataModel::addEvents);
|
2016-05-10 13:27:48 +02:00
|
|
|
|
2016-02-08 16:51:48 +01:00
|
|
|
connect(reader, &QmlProfilerFileReader::success, this, [this, reader]() {
|
2016-05-10 13:27:48 +02:00
|
|
|
d->traceTime->setTime(reader->traceStart(), reader->traceEnd());
|
2016-02-08 16:51:48 +01:00
|
|
|
setRecordedFeatures(reader->loadedFeatures());
|
|
|
|
delete reader;
|
|
|
|
acquiringDone();
|
|
|
|
}, Qt::QueuedConnection);
|
2013-08-08 13:28:08 +02:00
|
|
|
|
2016-05-10 13:27:48 +02:00
|
|
|
QFuture<void> result = Utils::runAsync([isQtd, file, reader] (QFutureInterface<void> &future) {
|
2016-02-08 16:51:48 +01:00
|
|
|
reader->setFuture(&future);
|
2016-05-10 13:27:48 +02:00
|
|
|
if (isQtd)
|
|
|
|
reader->loadQtd(file);
|
|
|
|
else
|
|
|
|
reader->loadQzt(file);
|
2015-02-17 14:25:24 +01:00
|
|
|
file->close();
|
|
|
|
file->deleteLater();
|
|
|
|
});
|
2013-08-08 13:28:08 +02:00
|
|
|
|
2015-02-17 14:25:24 +01:00
|
|
|
Core::ProgressManager::addTask(result, tr("Loading Trace Data"), Constants::TASK_LOAD);
|
2013-08-08 13:28:08 +02:00
|
|
|
}
|
|
|
|
|
2015-09-10 13:51:54 +02:00
|
|
|
void QmlProfilerModelManager::setState(QmlProfilerModelManager::State state)
|
2013-08-08 13:28:08 +02:00
|
|
|
{
|
2015-09-10 13:51:54 +02:00
|
|
|
// It's not an error, we are continuously calling "AcquiringData" for example
|
|
|
|
if (d->state == state)
|
|
|
|
return;
|
|
|
|
|
|
|
|
switch (state) {
|
|
|
|
case ClearingData:
|
|
|
|
QTC_ASSERT(d->state == Done || d->state == Empty || d->state == AcquiringData, /**/);
|
|
|
|
break;
|
|
|
|
case Empty:
|
|
|
|
// if it's not empty, complain but go on
|
|
|
|
QTC_ASSERT(isEmpty(), /**/);
|
|
|
|
break;
|
|
|
|
case AcquiringData:
|
|
|
|
// we're not supposed to receive new data while processing older data
|
|
|
|
QTC_ASSERT(d->state != ProcessingData, return);
|
|
|
|
break;
|
|
|
|
case ProcessingData:
|
|
|
|
QTC_ASSERT(d->state == AcquiringData, return);
|
|
|
|
break;
|
|
|
|
case Done:
|
|
|
|
QTC_ASSERT(d->state == ProcessingData || d->state == Empty, return);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
emit error(tr("Trying to set unknown state in events list."));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
d->state = state;
|
|
|
|
emit stateChanged();
|
2013-08-08 13:28:08 +02:00
|
|
|
}
|
|
|
|
|
2015-09-10 13:51:54 +02:00
|
|
|
QmlProfilerModelManager::State QmlProfilerModelManager::state() const
|
2013-08-08 13:28:08 +02:00
|
|
|
{
|
2015-09-10 13:51:54 +02:00
|
|
|
return d->state;
|
2013-08-08 13:28:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void QmlProfilerModelManager::clear()
|
|
|
|
{
|
2015-09-10 13:51:54 +02:00
|
|
|
setState(ClearingData);
|
2016-05-11 14:43:26 +02:00
|
|
|
d->numLoadedEvents = 0;
|
|
|
|
d->numFinishedFinalizers = 0;
|
2013-08-08 13:28:08 +02:00
|
|
|
d->model->clear();
|
|
|
|
d->traceTime->clear();
|
2014-11-03 14:19:26 +01:00
|
|
|
d->notesModel->clear();
|
2015-06-30 15:55:33 +02:00
|
|
|
setVisibleFeatures(0);
|
|
|
|
setRecordedFeatures(0);
|
2013-08-08 13:28:08 +02:00
|
|
|
|
2015-09-10 13:51:54 +02:00
|
|
|
setState(Empty);
|
2013-08-08 13:28:08 +02:00
|
|
|
}
|
|
|
|
|
2016-04-28 16:13:16 +02:00
|
|
|
void QmlProfilerModelManager::restrictToRange(qint64 startTime, qint64 endTime)
|
|
|
|
{
|
2016-07-06 11:34:15 +02:00
|
|
|
d->notesModel->saveData();
|
2016-07-06 11:34:49 +02:00
|
|
|
const QVector<QmlNote> notes = d->notesModel->notes();
|
|
|
|
d->notesModel->clear();
|
|
|
|
|
2016-06-02 10:20:52 +02:00
|
|
|
setState(ClearingData);
|
2016-04-28 16:13:16 +02:00
|
|
|
setVisibleFeatures(0);
|
|
|
|
|
|
|
|
startAcquiring();
|
|
|
|
d->model->replayEvents(startTime, endTime,
|
|
|
|
std::bind(&QmlProfilerModelManager::dispatch, this,
|
|
|
|
std::placeholders::_1, std::placeholders::_2));
|
2016-07-06 11:34:49 +02:00
|
|
|
d->notesModel->setNotes(notes);
|
2016-04-28 16:13:16 +02:00
|
|
|
d->traceTime->restrictToRange(startTime, endTime);
|
|
|
|
acquiringDone();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool QmlProfilerModelManager::isRestrictedToRange() const
|
|
|
|
{
|
|
|
|
return d->traceTime->isRestrictedToRange();
|
|
|
|
}
|
|
|
|
|
2016-04-28 16:02:54 +02:00
|
|
|
void QmlProfilerModelManager::startAcquiring()
|
2013-08-08 13:28:08 +02:00
|
|
|
{
|
2015-09-10 13:51:54 +02:00
|
|
|
setState(AcquiringData);
|
2013-08-08 13:28:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace QmlProfiler
|