2014-05-27 16:30:48 +02:00
|
|
|
/****************************************************************************
|
|
|
|
|
**
|
|
|
|
|
** Copyright (C) 2013 Digia Plc
|
|
|
|
|
** All rights reserved.
|
|
|
|
|
** For any questions to Digia, please use contact form at http://qt.digia.com <http://qt.digia.com/>
|
|
|
|
|
**
|
|
|
|
|
** This file is part of the Qt Enterprise Qt Quick Profiler Add-on.
|
|
|
|
|
**
|
|
|
|
|
** Licensees holding valid Qt Enterprise licenses may use this file in
|
|
|
|
|
** accordance with the Qt Enterprise License Agreement provided with the
|
|
|
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
|
|
|
** a written agreement between you and Digia.
|
|
|
|
|
**
|
|
|
|
|
** If you have questions regarding the use of this file, please use
|
|
|
|
|
** contact form at http://qt.digia.com <http://qt.digia.com/>
|
|
|
|
|
**
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
|
|
#include "memoryusagemodel.h"
|
|
|
|
|
#include "qmldebug/qmlprofilereventtypes.h"
|
|
|
|
|
#include "qmlprofiler/qmlprofilermodelmanager.h"
|
|
|
|
|
#include "qmlprofiler/sortedtimelinemodel.h"
|
2014-06-12 16:03:40 +02:00
|
|
|
#include "qmlprofiler/abstracttimelinemodel_p.h"
|
2014-05-27 16:30:48 +02:00
|
|
|
|
2014-06-24 11:53:47 +02:00
|
|
|
#include <QStack>
|
2014-05-27 16:30:48 +02:00
|
|
|
|
|
|
|
|
namespace QmlProfilerExtension {
|
|
|
|
|
namespace Internal {
|
|
|
|
|
|
|
|
|
|
using namespace QmlProfiler;
|
|
|
|
|
|
|
|
|
|
class MemoryUsageModel::MemoryUsageModelPrivate :
|
|
|
|
|
public SortedTimelineModel<MemoryAllocation,
|
2014-06-12 16:03:40 +02:00
|
|
|
AbstractTimelineModel::AbstractTimelineModelPrivate>
|
2014-05-27 16:30:48 +02:00
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
static QString memoryTypeName(int type);
|
|
|
|
|
|
|
|
|
|
qint64 maxSize;
|
|
|
|
|
private:
|
|
|
|
|
Q_DECLARE_PUBLIC(MemoryUsageModel)
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
MemoryUsageModel::MemoryUsageModel(QObject *parent)
|
2014-06-12 16:03:40 +02:00
|
|
|
: AbstractTimelineModel(new MemoryUsageModelPrivate(),
|
2014-05-27 16:30:48 +02:00
|
|
|
QLatin1String("MemoryUsageTimelineModel"),
|
|
|
|
|
QLatin1String("Memory Usage"), QmlDebug::MemoryAllocation,
|
|
|
|
|
QmlDebug::MaximumRangeType, parent)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int MemoryUsageModel::rowCount() const
|
|
|
|
|
{
|
|
|
|
|
return isEmpty() ? 1 : 3;
|
|
|
|
|
}
|
|
|
|
|
|
2014-06-24 11:53:19 +02:00
|
|
|
int MemoryUsageModel::rowMaxValue(int rowNumber) const
|
|
|
|
|
{
|
|
|
|
|
Q_D(const MemoryUsageModel);
|
|
|
|
|
Q_UNUSED(rowNumber);
|
|
|
|
|
return d->maxSize;
|
|
|
|
|
}
|
|
|
|
|
|
2014-05-27 16:30:48 +02:00
|
|
|
int MemoryUsageModel::getEventRow(int index) const
|
|
|
|
|
{
|
|
|
|
|
Q_D(const MemoryUsageModel);
|
|
|
|
|
QmlDebug::MemoryType type = d->range(index).type;
|
|
|
|
|
if (type == QmlDebug::HeapPage || type == QmlDebug::LargeItem)
|
|
|
|
|
return 1;
|
|
|
|
|
else
|
|
|
|
|
return 2;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int MemoryUsageModel::getEventId(int index) const
|
|
|
|
|
{
|
|
|
|
|
Q_D(const MemoryUsageModel);
|
|
|
|
|
return d->range(index).type;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QColor MemoryUsageModel::getColor(int index) const
|
|
|
|
|
{
|
|
|
|
|
return getEventColor(index);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
float MemoryUsageModel::getHeight(int index) const
|
|
|
|
|
{
|
|
|
|
|
Q_D(const MemoryUsageModel);
|
2014-06-24 11:53:19 +02:00
|
|
|
return qMin(1.0f, (float)d->range(index).size / (float)d->maxSize);
|
2014-05-27 16:30:48 +02:00
|
|
|
}
|
|
|
|
|
|
2014-06-24 11:53:47 +02:00
|
|
|
const QVariantMap MemoryUsageModel::getEventLocation(int index) const
|
|
|
|
|
{
|
|
|
|
|
static const QLatin1String file("file");
|
|
|
|
|
static const QLatin1String line("line");
|
|
|
|
|
static const QLatin1String column("column");
|
|
|
|
|
|
|
|
|
|
Q_D(const MemoryUsageModel);
|
|
|
|
|
QVariantMap result;
|
|
|
|
|
|
|
|
|
|
int originType = d->range(index).originTypeIndex;
|
|
|
|
|
if (originType > -1) {
|
|
|
|
|
const QmlDebug::QmlEventLocation &location =
|
|
|
|
|
d->modelManager->qmlModel()->getEventTypes().at(originType).location;
|
|
|
|
|
|
|
|
|
|
result.insert(file, location.filename);
|
|
|
|
|
result.insert(line, location.line);
|
|
|
|
|
result.insert(column, location.column);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2014-05-27 16:30:48 +02:00
|
|
|
const QVariantList MemoryUsageModel::getLabels() const
|
|
|
|
|
{
|
|
|
|
|
Q_D(const MemoryUsageModel);
|
|
|
|
|
QVariantList result;
|
|
|
|
|
|
|
|
|
|
if (d->expanded && !isEmpty()) {
|
|
|
|
|
{
|
|
|
|
|
QVariantMap element;
|
|
|
|
|
element.insert(QLatin1String("description"), QVariant(tr("Memory Allocation")));
|
|
|
|
|
|
|
|
|
|
element.insert(QLatin1String("id"), QVariant(QmlDebug::HeapPage));
|
|
|
|
|
result << element;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
QVariantMap element;
|
|
|
|
|
element.insert(QLatin1String("description"), QVariant(tr("Memory Usage")));
|
|
|
|
|
|
|
|
|
|
element.insert(QLatin1String("id"), QVariant(QmlDebug::SmallItem));
|
|
|
|
|
result << element;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const QVariantList MemoryUsageModel::getEventDetails(int index) const
|
|
|
|
|
{
|
|
|
|
|
Q_D(const MemoryUsageModel);
|
2014-07-11 10:09:08 +02:00
|
|
|
static QString title = QStringLiteral("title");
|
|
|
|
|
|
2014-05-27 16:30:48 +02:00
|
|
|
QVariantList result;
|
|
|
|
|
const MemoryUsageModelPrivate::Range *ev = &d->range(index);
|
|
|
|
|
|
2014-07-11 10:09:08 +02:00
|
|
|
QVariantMap res;
|
2014-07-11 11:01:47 +02:00
|
|
|
if (ev->allocated >= -ev->deallocated)
|
2014-07-11 10:09:08 +02:00
|
|
|
res.insert(title, tr("Memory Allocated"));
|
|
|
|
|
else
|
|
|
|
|
res.insert(title, tr("Memory Freed"));
|
|
|
|
|
result << res;
|
|
|
|
|
res.clear();
|
2014-05-27 16:30:48 +02:00
|
|
|
|
2014-07-11 10:09:08 +02:00
|
|
|
res.insert(tr("Total"), QVariant(QString::fromLatin1("%1 bytes").arg(ev->size)));
|
|
|
|
|
result << res;
|
|
|
|
|
res.clear();
|
2014-05-27 16:30:48 +02:00
|
|
|
|
2014-07-11 11:01:47 +02:00
|
|
|
if (ev->allocations > 0) {
|
|
|
|
|
res.insert(tr("Allocated"), QString::fromLatin1("%1 bytes").arg(ev->allocated));
|
|
|
|
|
res.insert(tr("Allocations"), QString::number(ev->allocations));
|
|
|
|
|
}
|
|
|
|
|
if (ev->deallocations > 0) {
|
|
|
|
|
res.insert(tr("Deallocated"), QString::fromLatin1("%1 bytes").arg(-ev->deallocated));
|
|
|
|
|
res.insert(tr("Deallocations"), QString::number(ev->deallocations));
|
|
|
|
|
}
|
2014-07-11 10:09:08 +02:00
|
|
|
res.insert(tr("Type"), QVariant(MemoryUsageModelPrivate::memoryTypeName(ev->type)));
|
2014-06-24 11:53:47 +02:00
|
|
|
if (ev->originTypeIndex != -1) {
|
2014-07-11 10:09:08 +02:00
|
|
|
res.insert(tr("Location"),
|
2014-06-24 11:53:47 +02:00
|
|
|
d->modelManager->qmlModel()->getEventTypes().at(ev->originTypeIndex).displayName);
|
|
|
|
|
}
|
2014-07-11 10:09:08 +02:00
|
|
|
result << res;
|
2014-05-27 16:30:48 +02:00
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2014-06-24 11:53:47 +02:00
|
|
|
struct RangeStackFrame {
|
|
|
|
|
RangeStackFrame() : originTypeIndex(-1), startTime(-1), endTime(-1) {}
|
|
|
|
|
RangeStackFrame(int originTypeIndex, qint64 startTime, qint64 endTime) :
|
|
|
|
|
originTypeIndex(originTypeIndex), startTime(startTime), endTime(endTime) {}
|
|
|
|
|
int originTypeIndex;
|
|
|
|
|
qint64 startTime;
|
|
|
|
|
qint64 endTime;
|
|
|
|
|
};
|
|
|
|
|
|
2014-05-27 16:30:48 +02:00
|
|
|
void MemoryUsageModel::loadData()
|
|
|
|
|
{
|
|
|
|
|
Q_D(MemoryUsageModel);
|
|
|
|
|
clear();
|
|
|
|
|
QmlProfilerDataModel *simpleModel = d->modelManager->qmlModel();
|
|
|
|
|
if (simpleModel->isEmpty())
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
qint64 currentSize = 0;
|
|
|
|
|
qint64 currentUsage = 0;
|
|
|
|
|
int currentUsageIndex = -1;
|
|
|
|
|
int currentJSHeapIndex = -1;
|
2014-06-24 11:53:47 +02:00
|
|
|
|
|
|
|
|
QStack<RangeStackFrame> rangeStack;
|
2014-07-11 11:01:47 +02:00
|
|
|
MemoryAllocation dummy;
|
2014-06-24 11:53:47 +02:00
|
|
|
|
2014-06-13 16:56:46 +02:00
|
|
|
const QVector<QmlProfilerDataModel::QmlEventTypeData> &types = simpleModel->getEventTypes();
|
2014-05-27 16:30:48 +02:00
|
|
|
foreach (const QmlProfilerDataModel::QmlEventData &event, simpleModel->getEvents()) {
|
2014-06-13 16:56:46 +02:00
|
|
|
const QmlProfilerDataModel::QmlEventTypeData &type = types[event.typeIndex];
|
2014-06-24 11:53:47 +02:00
|
|
|
while (!rangeStack.empty() && rangeStack.top().endTime < event.startTime)
|
|
|
|
|
rangeStack.pop();
|
|
|
|
|
if (!eventAccepted(type)) {
|
|
|
|
|
if (type.rangeType != QmlDebug::MaximumRangeType) {
|
|
|
|
|
rangeStack.push(RangeStackFrame(event.typeIndex, event.startTime,
|
|
|
|
|
event.startTime + event.duration));
|
|
|
|
|
}
|
2014-05-27 16:30:48 +02:00
|
|
|
continue;
|
2014-06-24 11:53:47 +02:00
|
|
|
}
|
2014-05-27 16:30:48 +02:00
|
|
|
|
2014-06-13 16:56:46 +02:00
|
|
|
if (type.detailType == QmlDebug::SmallItem || type.detailType == QmlDebug::LargeItem) {
|
2014-06-24 11:53:47 +02:00
|
|
|
MemoryAllocation &last = currentUsageIndex > -1 ? d->data(currentUsageIndex) : dummy;
|
2014-07-11 11:01:47 +02:00
|
|
|
if (!rangeStack.empty() && type.detailType == last.type &&
|
|
|
|
|
last.originTypeIndex == rangeStack.top().originTypeIndex &&
|
|
|
|
|
rangeStack.top().startTime < d->range(currentUsageIndex).start) {
|
|
|
|
|
last.update(event.numericData1);
|
|
|
|
|
currentUsage = last.size;
|
2014-06-24 11:53:47 +02:00
|
|
|
} else {
|
2014-07-11 11:01:47 +02:00
|
|
|
MemoryAllocation allocation(QmlDebug::SmallItem, currentUsage,
|
|
|
|
|
rangeStack.empty() ? -1 : rangeStack.top().originTypeIndex);
|
|
|
|
|
allocation.update(event.numericData1);
|
|
|
|
|
currentUsage = allocation.size;
|
|
|
|
|
|
2014-06-24 11:53:47 +02:00
|
|
|
if (currentUsageIndex != -1) {
|
|
|
|
|
d->insertEnd(currentUsageIndex,
|
|
|
|
|
event.startTime - d->range(currentUsageIndex).start - 1);
|
|
|
|
|
}
|
|
|
|
|
currentUsageIndex = d->insertStart(event.startTime, allocation);
|
2014-05-27 16:30:48 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-06-13 16:56:46 +02:00
|
|
|
if (type.detailType == QmlDebug::HeapPage || type.detailType == QmlDebug::LargeItem) {
|
2014-06-24 11:53:47 +02:00
|
|
|
MemoryAllocation &last = currentJSHeapIndex > -1 ? d->data(currentJSHeapIndex) : dummy;
|
2014-07-11 11:01:47 +02:00
|
|
|
if (!rangeStack.empty() && type.detailType == last.type &&
|
|
|
|
|
last.originTypeIndex == rangeStack.top().originTypeIndex &&
|
|
|
|
|
rangeStack.top().startTime < d->range(currentJSHeapIndex).start) {
|
|
|
|
|
last.update(event.numericData1);
|
|
|
|
|
currentSize = last.size;
|
2014-06-24 11:53:47 +02:00
|
|
|
} else {
|
2014-07-11 11:01:47 +02:00
|
|
|
MemoryAllocation allocation((QmlDebug::MemoryType)type.detailType, currentSize,
|
|
|
|
|
rangeStack.empty() ? -1 : rangeStack.top().originTypeIndex);
|
|
|
|
|
allocation.update(event.numericData1);
|
|
|
|
|
currentSize = allocation.size;
|
2014-06-24 11:53:47 +02:00
|
|
|
|
|
|
|
|
if (currentSize > d->maxSize)
|
|
|
|
|
d->maxSize = currentSize;
|
|
|
|
|
if (currentJSHeapIndex != -1)
|
|
|
|
|
d->insertEnd(currentJSHeapIndex,
|
|
|
|
|
event.startTime - d->range(currentJSHeapIndex).start - 1);
|
|
|
|
|
currentJSHeapIndex = d->insertStart(event.startTime, allocation);
|
|
|
|
|
}
|
2014-05-27 16:30:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
d->modelManager->modelProxyCountUpdated(d->modelId, d->count(), simpleModel->getEvents().count());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (currentJSHeapIndex != -1)
|
2014-07-08 12:33:46 +02:00
|
|
|
d->insertEnd(currentJSHeapIndex, traceEndTime() -
|
2014-05-27 16:30:48 +02:00
|
|
|
d->range(currentJSHeapIndex).start - 1);
|
|
|
|
|
if (currentUsageIndex != -1)
|
2014-07-08 12:33:46 +02:00
|
|
|
d->insertEnd(currentUsageIndex, traceEndTime() -
|
2014-05-27 16:30:48 +02:00
|
|
|
d->range(currentUsageIndex).start - 1);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
d->computeNesting();
|
|
|
|
|
d->modelManager->modelProxyCountUpdated(d->modelId, 1, 1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void MemoryUsageModel::clear()
|
|
|
|
|
{
|
|
|
|
|
Q_D(MemoryUsageModel);
|
|
|
|
|
d->SortedTimelineModel::clear();
|
|
|
|
|
d->expanded = false;
|
|
|
|
|
d->maxSize = 1;
|
|
|
|
|
|
|
|
|
|
d->modelManager->modelProxyCountUpdated(d->modelId, 0, 1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString MemoryUsageModel::MemoryUsageModelPrivate::memoryTypeName(int type)
|
|
|
|
|
{
|
|
|
|
|
switch (type) {
|
|
|
|
|
case QmlDebug::HeapPage: return tr("Heap Allocation");
|
|
|
|
|
case QmlDebug::LargeItem: return tr("Large Item Allocation");
|
|
|
|
|
case QmlDebug::SmallItem: return tr("Heap Usage");
|
|
|
|
|
case QmlDebug::MaximumMemoryType: return tr("Total");
|
|
|
|
|
default: return tr("Unknown");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-07-11 11:01:47 +02:00
|
|
|
MemoryUsageModel::MemoryAllocation::MemoryAllocation(QmlDebug::MemoryType type, qint64 baseAmount,
|
|
|
|
|
int originTypeIndex) :
|
|
|
|
|
type(type), size(baseAmount), allocated(0), deallocated(0), allocations(0), deallocations(0),
|
|
|
|
|
originTypeIndex(originTypeIndex)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void MemoryUsageModel::MemoryAllocation::update(qint64 amount)
|
|
|
|
|
{
|
|
|
|
|
size += amount;
|
|
|
|
|
if (amount < 0) {
|
|
|
|
|
deallocated += amount;
|
|
|
|
|
++deallocations;
|
|
|
|
|
} else {
|
|
|
|
|
allocated += amount;
|
|
|
|
|
++allocations;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-05-27 16:30:48 +02:00
|
|
|
|
|
|
|
|
} // namespace Internal
|
|
|
|
|
} // namespace QmlProfilerExtension
|