forked from qt-creator/qt-creator
... in turn, make its members private, so that we don't accidentally change them. Change-Id: Ibc65b406ee341d33f69647ed1b19e1e34f5cd535 Reviewed-by: Christian Kandeler <christian.kandeler@theqtcompany.com>
247 lines
7.6 KiB
C++
247 lines
7.6 KiB
C++
/****************************************************************************
|
|
**
|
|
** 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 "qmlprofilerrangemodel.h"
|
|
#include "qmlprofilermodelmanager.h"
|
|
#include "qmlprofilerdatamodel.h"
|
|
#include "qmlprofilerbindingloopsrenderpass.h"
|
|
|
|
#include "timeline/timelinenotesrenderpass.h"
|
|
#include "timeline/timelineitemsrenderpass.h"
|
|
#include "timeline/timelineselectionrenderpass.h"
|
|
|
|
#include <QCoreApplication>
|
|
#include <QVector>
|
|
#include <QHash>
|
|
#include <QUrl>
|
|
#include <QString>
|
|
#include <QStack>
|
|
|
|
namespace QmlProfiler {
|
|
namespace Internal {
|
|
|
|
QmlProfilerRangeModel::QmlProfilerRangeModel(QmlProfilerModelManager *manager, RangeType range,
|
|
QObject *parent) :
|
|
QmlProfilerTimelineModel(manager, MaximumMessage, range, featureFromRangeType(range), parent)
|
|
{
|
|
m_expandedRowTypes << -1;
|
|
}
|
|
|
|
void QmlProfilerRangeModel::clear()
|
|
{
|
|
m_expandedRowTypes.clear();
|
|
m_expandedRowTypes << -1;
|
|
m_data.clear();
|
|
m_stack.clear();
|
|
QmlProfilerTimelineModel::clear();
|
|
}
|
|
|
|
bool QmlProfilerRangeModel::supportsBindingLoops() const
|
|
{
|
|
return rangeType() == Binding || rangeType() == HandlingSignal;
|
|
}
|
|
|
|
void QmlProfilerRangeModel::loadEvent(const QmlEvent &event, const QmlEventType &type)
|
|
{
|
|
Q_UNUSED(type);
|
|
// store starttime-based instance
|
|
if (event.rangeStage() == RangeStart) {
|
|
int index = insertStart(event.timestamp(), event.typeIndex());
|
|
m_stack.append(index);
|
|
m_data.insert(index, QmlRangeEventStartInstance());
|
|
} else if (event.rangeStage() == RangeEnd) {
|
|
int index = m_stack.pop();
|
|
insertEnd(index, event.timestamp() - startTime(index));
|
|
}
|
|
}
|
|
|
|
void QmlProfilerRangeModel::finalize()
|
|
{
|
|
// compute range nesting
|
|
computeNesting();
|
|
|
|
// compute nestingLevel - nonexpanded
|
|
computeNestingContracted();
|
|
|
|
// compute nestingLevel - expanded
|
|
computeExpandedLevels();
|
|
|
|
if (supportsBindingLoops())
|
|
findBindingLoops();
|
|
}
|
|
|
|
void QmlProfilerRangeModel::computeNestingContracted()
|
|
{
|
|
int i;
|
|
int eventCount = count();
|
|
|
|
int nestingLevels = Constants::QML_MIN_LEVEL;
|
|
int collapsedRowCount = nestingLevels + 1;
|
|
QVector<qint64> nestingEndTimes;
|
|
nestingEndTimes.fill(0, nestingLevels + 1);
|
|
|
|
for (i = 0; i < eventCount; i++) {
|
|
qint64 st = startTime(i);
|
|
|
|
// per type
|
|
if (nestingEndTimes[nestingLevels] > st) {
|
|
if (++nestingLevels == nestingEndTimes.size())
|
|
nestingEndTimes << 0;
|
|
if (nestingLevels == collapsedRowCount)
|
|
++collapsedRowCount;
|
|
} else {
|
|
while (nestingLevels > Constants::QML_MIN_LEVEL &&
|
|
nestingEndTimes[nestingLevels-1] <= st)
|
|
nestingLevels--;
|
|
}
|
|
nestingEndTimes[nestingLevels] = st + duration(i);
|
|
|
|
m_data[i].displayRowCollapsed = nestingLevels;
|
|
}
|
|
setCollapsedRowCount(collapsedRowCount);
|
|
}
|
|
|
|
void QmlProfilerRangeModel::computeExpandedLevels()
|
|
{
|
|
QHash<int, int> eventRow;
|
|
int eventCount = count();
|
|
for (int i = 0; i < eventCount; i++) {
|
|
int eventTypeId = typeId(i);
|
|
if (!eventRow.contains(eventTypeId)) {
|
|
eventRow[eventTypeId] = m_expandedRowTypes.size();
|
|
m_expandedRowTypes << eventTypeId;
|
|
}
|
|
m_data[i].displayRowExpanded = eventRow[eventTypeId];
|
|
}
|
|
setExpandedRowCount(m_expandedRowTypes.size());
|
|
}
|
|
|
|
void QmlProfilerRangeModel::findBindingLoops()
|
|
{
|
|
typedef QPair<int, int> CallStackEntry;
|
|
QStack<CallStackEntry> callStack;
|
|
|
|
for (int i = 0; i < count(); ++i) {
|
|
int potentialParent = callStack.isEmpty() ? -1 : callStack.top().second;
|
|
|
|
while (potentialParent != -1 && !(endTime(potentialParent) > startTime(i))) {
|
|
callStack.pop();
|
|
potentialParent = callStack.isEmpty() ? -1 : callStack.top().second;
|
|
}
|
|
|
|
// check whether event is already in stack
|
|
for (int ii = 0; ii < callStack.size(); ++ii) {
|
|
if (callStack.at(ii).first == typeId(i)) {
|
|
m_data[i].bindingLoopHead = callStack.at(ii).second;
|
|
break;
|
|
}
|
|
}
|
|
|
|
CallStackEntry newEntry(typeId(i), i);
|
|
callStack.push(newEntry);
|
|
}
|
|
|
|
}
|
|
|
|
int QmlProfilerRangeModel::expandedRow(int index) const
|
|
{
|
|
return m_data[index].displayRowExpanded;
|
|
}
|
|
|
|
int QmlProfilerRangeModel::collapsedRow(int index) const
|
|
{
|
|
return m_data[index].displayRowCollapsed;
|
|
}
|
|
|
|
int QmlProfilerRangeModel::bindingLoopDest(int index) const
|
|
{
|
|
return m_data[index].bindingLoopHead;
|
|
}
|
|
|
|
QColor QmlProfilerRangeModel::color(int index) const
|
|
{
|
|
return colorBySelectionId(index);
|
|
}
|
|
|
|
QVariantList QmlProfilerRangeModel::labels() const
|
|
{
|
|
QVariantList result;
|
|
|
|
const QVector<QmlEventType> &types = modelManager()->qmlModel()->eventTypes();
|
|
for (int i = 1; i < expandedRowCount(); i++) { // Ignore the -1 for the first row
|
|
QVariantMap element;
|
|
int typeId = m_expandedRowTypes[i];
|
|
element.insert(QLatin1String("displayName"), QVariant(types[typeId].displayName()));
|
|
element.insert(QLatin1String("description"), QVariant(types[typeId].data()));
|
|
element.insert(QLatin1String("id"), QVariant(typeId));
|
|
result << element;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
QVariantMap QmlProfilerRangeModel::details(int index) const
|
|
{
|
|
QVariantMap result;
|
|
int id = selectionId(index);
|
|
const QVector<QmlEventType> &types = modelManager()->qmlModel()->eventTypes();
|
|
|
|
result.insert(QStringLiteral("displayName"),
|
|
tr(QmlProfilerModelManager::featureName(mainFeature())));
|
|
result.insert(tr("Duration"), QmlProfilerDataModel::formatTime(duration(index)));
|
|
|
|
result.insert(tr("Details"), types[id].data());
|
|
result.insert(tr("Location"), types[id].displayName());
|
|
return result;
|
|
}
|
|
|
|
QVariantMap QmlProfilerRangeModel::location(int index) const
|
|
{
|
|
return locationFromTypeId(index);
|
|
}
|
|
|
|
int QmlProfilerRangeModel::typeId(int index) const
|
|
{
|
|
return selectionId(index);
|
|
}
|
|
|
|
QList<const Timeline::TimelineRenderPass *> QmlProfilerRangeModel::supportedRenderPasses() const
|
|
{
|
|
if (supportsBindingLoops()) {
|
|
QList<const Timeline::TimelineRenderPass *> passes;
|
|
passes << Timeline::TimelineItemsRenderPass::instance()
|
|
<< QmlProfilerBindingLoopsRenderPass::instance()
|
|
<< Timeline::TimelineSelectionRenderPass::instance()
|
|
<< Timeline::TimelineNotesRenderPass::instance();
|
|
return passes;
|
|
} else {
|
|
return QmlProfilerTimelineModel::supportedRenderPasses();
|
|
}
|
|
|
|
}
|
|
|
|
} // namespace Internal
|
|
} // namespaec QmlProfiler
|