forked from qt-creator/qt-creator
Since we also license under GPL-3.0 WITH Qt-GPL-exception-1.0,
this applies only to a hypothetical newer version of GPL, that doesn't
exist yet. If such a version emerges, we can still decide to relicense...
While at it, replace (deprecated) GPL-3.0 with more explicit GPL-3.0-only
Change was done by running
find . -type f -exec perl -pi -e "s/LicenseRef-Qt-Commercial OR GPL-3.0\+ OR GPL-3.0 WITH Qt-GPL-exception-1.0/LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0/g" {} \;
Change-Id: I5097e6ce8d10233993ee30d7e25120e2659eb10b
Reviewed-by: Eike Ziller <eike.ziller@qt.io>
242 lines
7.0 KiB
C++
242 lines
7.0 KiB
C++
// Copyright (C) 2016 The Qt Company Ltd.
|
|
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
|
|
|
|
#include "qmlprofilerbindingloopsrenderpass.h"
|
|
#include "qmlprofilermodelmanager.h"
|
|
#include "qmlprofilerrangemodel.h"
|
|
#include "qmlprofilertr.h"
|
|
|
|
#include <tracing/timelinenotesrenderpass.h>
|
|
#include <tracing/timelineitemsrenderpass.h>
|
|
#include <tracing/timelineselectionrenderpass.h>
|
|
#include <tracing/timelineformattime.h>
|
|
|
|
#include <QCoreApplication>
|
|
#include <QVector>
|
|
#include <QHash>
|
|
#include <QUrl>
|
|
#include <QString>
|
|
#include <QStack>
|
|
|
|
namespace QmlProfiler {
|
|
namespace Internal {
|
|
|
|
QmlProfilerRangeModel::QmlProfilerRangeModel(QmlProfilerModelManager *manager, RangeType range,
|
|
Timeline::TimelineModelAggregator *parent) :
|
|
QmlProfilerTimelineModel(manager, UndefinedMessage, 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, Item());
|
|
} else if (event.rangeStage() == RangeEnd) {
|
|
if (!m_stack.isEmpty()) {
|
|
int index = m_stack.pop();
|
|
insertEnd(index, event.timestamp() - startTime(index));
|
|
} else {
|
|
qWarning() << "Received inconsistent trace data from application.";
|
|
}
|
|
}
|
|
}
|
|
|
|
void QmlProfilerRangeModel::finalize()
|
|
{
|
|
if (!m_stack.isEmpty()) {
|
|
qWarning() << "End times for some events are missing.";
|
|
const qint64 endTime = modelManager()->traceEnd();
|
|
do {
|
|
int index = m_stack.pop();
|
|
insertEnd(index, endTime - startTime(index));
|
|
} while (!m_stack.isEmpty());
|
|
}
|
|
|
|
// compute range nesting
|
|
computeNesting();
|
|
|
|
// compute nestingLevel - nonexpanded
|
|
computeNestingContracted();
|
|
|
|
// compute nestingLevel - expanded
|
|
computeExpandedLevels();
|
|
|
|
if (supportsBindingLoops())
|
|
findBindingLoops();
|
|
|
|
QmlProfilerTimelineModel::finalize();
|
|
}
|
|
|
|
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()
|
|
{
|
|
using CallStackEntry = QPair<int, int>;
|
|
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;
|
|
}
|
|
|
|
QRgb QmlProfilerRangeModel::color(int index) const
|
|
{
|
|
return colorBySelectionId(index);
|
|
}
|
|
|
|
QVariantList QmlProfilerRangeModel::labels() const
|
|
{
|
|
QVariantList result;
|
|
|
|
const QmlProfilerModelManager *manager = modelManager();
|
|
for (int i = 1; i < expandedRowCount(); i++) { // Ignore the -1 for the first row
|
|
QVariantMap element;
|
|
const int typeId = m_expandedRowTypes[i];
|
|
const QmlEventType &type = manager->eventType(typeId);
|
|
element.insert(QLatin1String("displayName"), type.displayName());
|
|
element.insert(QLatin1String("description"), type.data());
|
|
element.insert(QLatin1String("id"), typeId);
|
|
result << element;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
QVariantMap QmlProfilerRangeModel::details(int index) const
|
|
{
|
|
QVariantMap result;
|
|
int id = selectionId(index);
|
|
|
|
result.insert(QStringLiteral("displayName"),
|
|
Tr::tr(QmlProfilerModelManager::featureName(mainFeature())));
|
|
result.insert(Tr::tr("Duration"), Timeline::formatTime(duration(index)));
|
|
|
|
const QmlEventType &type = modelManager()->eventType(id);
|
|
result.insert(Tr::tr("Details"), type.data());
|
|
result.insert(Tr::tr("Location"), type.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
|