/**************************************************************************** ** ** Copyright (C) 2013 Digia Plc ** All rights reserved. ** For any questions to Digia, please use contact form at 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 ** ****************************************************************************/ #include "memoryusagemodel.h" #include "qmldebug/qmlprofilereventtypes.h" #include "qmlprofiler/qmlprofilermodelmanager.h" #include namespace QmlProfilerExtension { namespace Internal { using namespace QmlProfiler; MemoryUsageModel::MemoryUsageModel(QmlProfilerModelManager *manager, QObject *parent) : QmlProfilerTimelineModel(manager, tr(QmlProfilerModelManager::featureName(QmlDebug::ProfileMemory)), QmlDebug::MemoryAllocation, QmlDebug::MaximumRangeType, parent) { m_maxSize = 1; announceFeatures((1 << QmlDebug::ProfileMemory) | QmlDebug::Constants::QML_JS_RANGE_FEATURES); } int MemoryUsageModel::rowMaxValue(int rowNumber) const { Q_UNUSED(rowNumber); return m_maxSize; } int MemoryUsageModel::expandedRow(int index) const { int type = selectionId(index); return (type == QmlDebug::HeapPage || type == QmlDebug::LargeItem) ? 1 : 2; } int MemoryUsageModel::collapsedRow(int index) const { return expandedRow(index); } int MemoryUsageModel::typeId(int index) const { return m_data[index].typeId; } QColor MemoryUsageModel::color(int index) const { return colorBySelectionId(index); } float MemoryUsageModel::relativeHeight(int index) const { return qMin(1.0f, (float)m_data[index].size / (float)m_maxSize); } QVariantMap MemoryUsageModel::location(int index) const { static const QLatin1String file("file"); static const QLatin1String line("line"); static const QLatin1String column("column"); QVariantMap result; int originType = m_data[index].originTypeIndex; if (originType > -1) { const QmlDebug::QmlEventLocation &location = modelManager()->qmlModel()->getEventTypes().at(originType).location; result.insert(file, location.filename); result.insert(line, location.line); result.insert(column, location.column); } return result; } QVariantList MemoryUsageModel::labels() const { QVariantList result; QVariantMap element; element.insert(QLatin1String("description"), QVariant(tr("Memory Allocation"))); element.insert(QLatin1String("id"), QVariant(QmlDebug::HeapPage)); result << element; element.clear(); element.insert(QLatin1String("description"), QVariant(tr("Memory Usage"))); element.insert(QLatin1String("id"), QVariant(QmlDebug::SmallItem)); result << element; return result; } QVariantMap MemoryUsageModel::details(int index) const { QVariantMap result; const MemoryAllocation *ev = &m_data[index]; if (ev->allocated >= -ev->deallocated) result.insert(QLatin1String("displayName"), tr("Memory Allocated")); else result.insert(QLatin1String("displayName"), tr("Memory Freed")); result.insert(tr("Total"), QString::fromLatin1("%1 bytes").arg(ev->size)); if (ev->allocations > 0) { result.insert(tr("Allocated"), QString::fromLatin1("%1 bytes").arg(ev->allocated)); result.insert(tr("Allocations"), QString::number(ev->allocations)); } if (ev->deallocations > 0) { result.insert(tr("Deallocated"), QString::fromLatin1("%1 bytes").arg(-ev->deallocated)); result.insert(tr("Deallocations"), QString::number(ev->deallocations)); } result.insert(tr("Type"), QVariant(memoryTypeName(selectionId(index)))); if (ev->originTypeIndex != -1) { result.insert(tr("Location"), modelManager()->qmlModel()->getEventTypes().at(ev->originTypeIndex).displayName); } return result; } 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; }; void MemoryUsageModel::loadData() { clear(); QmlProfilerDataModel *simpleModel = modelManager()->qmlModel(); if (simpleModel->isEmpty()) return; qint64 currentSize = 0; qint64 currentUsage = 0; int currentUsageIndex = -1; int currentJSHeapIndex = -1; QStack rangeStack; const QVector &types = simpleModel->getEventTypes(); foreach (const QmlProfilerDataModel::QmlEventData &event, simpleModel->getEvents()) { const QmlProfilerDataModel::QmlEventTypeData &type = types[event.typeIndex]; while (!rangeStack.empty() && rangeStack.top().endTime < event.startTime) rangeStack.pop(); if (!accepted(type)) { if (type.rangeType != QmlDebug::MaximumRangeType) { rangeStack.push(RangeStackFrame(event.typeIndex, event.startTime, event.startTime + event.duration)); } continue; } if (type.detailType == QmlDebug::SmallItem || type.detailType == QmlDebug::LargeItem) { if (!rangeStack.empty() && currentUsageIndex > -1 && type.detailType == selectionId(currentUsageIndex) && m_data[currentUsageIndex].originTypeIndex == rangeStack.top().originTypeIndex && rangeStack.top().startTime < startTime(currentUsageIndex)) { m_data[currentUsageIndex].update(event.numericData1); currentUsage = m_data[currentUsageIndex].size; } else { MemoryAllocation allocation(event.typeIndex, currentUsage, rangeStack.empty() ? -1 : rangeStack.top().originTypeIndex); allocation.update(event.numericData1); currentUsage = allocation.size; if (currentUsageIndex != -1) { insertEnd(currentUsageIndex, event.startTime - startTime(currentUsageIndex) - 1); } currentUsageIndex = insertStart(event.startTime, QmlDebug::SmallItem); m_data.insert(currentUsageIndex, allocation); } } if (type.detailType == QmlDebug::HeapPage || type.detailType == QmlDebug::LargeItem) { if (!rangeStack.empty() && currentJSHeapIndex > -1 && type.detailType == selectionId(currentJSHeapIndex) && m_data[currentJSHeapIndex].originTypeIndex == rangeStack.top().originTypeIndex && rangeStack.top().startTime < startTime(currentJSHeapIndex)) { m_data[currentJSHeapIndex].update(event.numericData1); currentSize = m_data[currentJSHeapIndex].size; } else { MemoryAllocation allocation(event.typeIndex, currentSize, rangeStack.empty() ? -1 : rangeStack.top().originTypeIndex); allocation.update(event.numericData1); currentSize = allocation.size; if (currentSize > m_maxSize) m_maxSize = currentSize; if (currentJSHeapIndex != -1) insertEnd(currentJSHeapIndex, event.startTime - startTime(currentJSHeapIndex) - 1); currentJSHeapIndex = insertStart(event.startTime, type.detailType); m_data.insert(currentJSHeapIndex, allocation); } } updateProgress(count(), simpleModel->getEvents().count()); } if (currentJSHeapIndex != -1) insertEnd(currentJSHeapIndex, modelManager()->traceTime()->endTime() - startTime(currentJSHeapIndex) - 1); if (currentUsageIndex != -1) insertEnd(currentUsageIndex, modelManager()->traceTime()->endTime() - startTime(currentUsageIndex) - 1); computeNesting(); setExpandedRowCount(3); setCollapsedRowCount(3); updateProgress(1, 1); } void MemoryUsageModel::clear() { m_data.clear(); m_maxSize = 1; QmlProfilerTimelineModel::clear(); } QString MemoryUsageModel::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"); } } MemoryUsageModel::MemoryAllocation::MemoryAllocation(int type, qint64 baseAmount, int originTypeIndex) : typeId(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; } } } // namespace Internal } // namespace QmlProfilerExtension