7#include <private/qobject_p.h>
9#include <QtCore/qfile.h>
10#include <QtCore/qqueue.h>
11#include <QtCore/qregularexpression.h>
12#include <QtCore/qurl.h>
13#include <QtCore/qxmlstream.h>
14#include <QtCore/qxpfunctional.h>
20using namespace Qt::StringLiterals;
33Q_STATIC_ASSERT(
sizeof(RANGE_TYPE_STRINGS) == MaximumRangeType *
sizeof(
const char *));
54 Q_DECLARE_PUBLIC(QQmlProfilerQtdWriter)
74 bool isEmpty()
const {
return events.isEmpty(); }
82 QQmlProfilerEventReceiver(*
new QQmlProfilerQtdWriterPrivate, parent)
90 Q_D(QQmlProfilerQtdWriter);
93 d->traceStartTime = std::numeric_limits<qint64>::max();
94 d->traceEndTime = std::numeric_limits<qint64>::min();
96 d->qmlMeasuredTime = 0;
97 d->setState(QQmlProfilerQtdWriterPrivate::Empty);
102 QQmlProfilerEventReceiver::startTrace(time, engineIds);
104 Q_D(QQmlProfilerQtdWriter);
105 if (time < d->traceStartTime)
106 d->traceStartTime = time;
111 QQmlProfilerEventReceiver::endTrace(time, engineIds);
113 Q_D(QQmlProfilerQtdWriter);
114 if (time > d->traceEndTime)
115 d->traceEndTime = time;
124 return QString::number(type);
132 return QString::number(type);
137 Q_D(QQmlProfilerQtdWriter);
138 d->setState(QQmlProfilerQtdWriterPrivate::AcquiringData);
139 d->events.append(event);
144 QQmlProfilerEventType newType = type;
148 if (!type.data().isEmpty()) {
149 details = type.data().simplified();
150 QRegularExpression rewrite(QStringLiteral(
"^\\(function \\$(\\w+)\\(\\) \\{ (return |)(.+) \\}\\)$"));
151 QRegularExpressionMatch match = rewrite.match(details);
152 if (match.hasMatch()) {
153 details = match.captured(1) +QLatin1String(
": ") + match.captured(3);
155 if (details.startsWith(QLatin1String(
"file://")))
156 details = details.mid(details.lastIndexOf(QLatin1Char(
'/')) + 1);
159 newType.setData(details);
162 switch (type.message()) {
164 switch (type.detailType()) {
167 displayName = QString::fromLatin1(
"Input:%1").arg(type.detailType());
170 displayName = QString::fromLatin1(
"AnimationFrame");
173 displayName = QString::fromLatin1(
"Unknown");
184 case PixmapCacheEvent: {
185 const QString filePath = QUrl(type.location().filename()).path();
186 displayName = QStringView{filePath}.mid(filePath.lastIndexOf(QLatin1Char(
'/')) + 1)
187 + QLatin1Char(
':') + QString::number(type.detailType());
190 case SceneGraphFrame:
191 displayName = QString::fromLatin1(
"SceneGraph:%1").arg(type.detailType());
193 case MemoryAllocation:
194 displayName = QString::fromLatin1(
"MemoryAllocation:%1").arg(type.detailType());
197 displayName = QString::fromLatin1(
"DebugMessage:%1").arg(type.detailType());
200 displayName = QString::fromLatin1(
"Quick3DFrame:%1").arg(type.detailType());
202 case MaximumMessage: {
203 const QQmlProfilerEventLocation eventLocation = type.location();
205 if (eventLocation.filename().isEmpty()) {
206 displayName = QString::fromLatin1(
"Unknown");
208 const QString filePath = QUrl(eventLocation.filename()).path();
209 displayName = QStringView{filePath}.mid(
210 filePath.lastIndexOf(QLatin1Char(
'/')) + 1) +
211 QLatin1Char(
':') + QString::number(eventLocation.line());
217 newType.setDisplayName(displayName);
218 d_func()->eventTypes.append(newType);
224 qint64 level0Start = -1;
227 for (
const QQmlProfilerEvent &event : std::as_const(events)) {
228 const QQmlProfilerEventType &type = eventTypes.at(event.typeIndex());
229 if (type.message() != MaximumMessage)
232 switch (type.rangeType()) {
238 switch (event.rangeStage()) {
241 level0Start = event.timestamp();
245 qmlMeasuredTime += event.timestamp() - level0Start;
259 return t1.timestamp() < t2.timestamp();
264 if (events.size() < 2)
269 QList<QQmlProfilerEvent>::iterator itFrom = events.end() - 2;
270 QList<QQmlProfilerEvent>::iterator itTo = events.end() - 1;
272 while (itFrom != events.begin() && itTo != events.begin()) {
274 while (itFrom != events.begin() && itTo->timestamp() > itFrom->timestamp()) {
280 if (itFrom == events.begin())
284 while (itFrom != events.begin() && itTo->timestamp() <= itFrom->timestamp())
287 if (itTo->timestamp() <= itFrom->timestamp())
288 std::sort(itFrom, itTo + 1, compareStartTimes);
290 std::sort(itFrom + 1, itTo + 1, compareStartTimes);
300 Q_D(QQmlProfilerQtdWriter);
301 d->setState(QQmlProfilerQtdWriterPrivate::ProcessingData);
304 d->setState(QQmlProfilerQtdWriterPrivate::Done);
305 QQmlProfilerEventReceiver::complete(maximumTime);
313 if (!filename.isEmpty()) {
314 file.setFileName(filename);
315 if (!file.open(QIODevice::WriteOnly)) {
316 error = QQmlProfilerQtdWriter::tr(
"Could not open %1 for writing").arg(filename);
320 if (!file.open(stdout, QIODevice::WriteOnly)) {
321 error = QQmlProfilerQtdWriter::tr(
"Could not open stdout for writing");
326 stream.setDevice(&file);
327 stream.setAutoFormatting(
true);
328 stream.writeStartDocument();
334 stream.writeEndDocument();
338 template<
typename Number>
341 stream.writeAttribute(QLatin1String(name), QString::number(number));
346 stream.writeAttribute(QLatin1String(name), QLatin1String(value));
349 void writeAttribute(
const char *name,
const QQmlProfilerEvent &event,
int i,
bool printZero =
true)
351 const qint64 number = event.number<qint64>(i);
352 if (printZero || number != 0)
353 writeAttribute(name, number);
356 template<
typename Number>
359 writeTextElement(name, QString::number(number));
364 stream.writeTextElement(QLatin1String(name), QLatin1String(value));
369 stream.writeTextElement(QLatin1String(name), value);
374 stream.writeStartElement(QLatin1String(name));
379 stream.writeEndElement();
384 QXmlStreamWriter stream;
391 qxp::function_ref<
void(
const QQmlProfilerEvent &, qint64)> &&sendEvent)
399 void handleRangeEvent(
const QQmlProfilerEvent &event,
const QQmlProfilerEventType &type);
404 const qxp::function_ref<
void(
const QQmlProfilerEvent &, qint64)> sendEvent;
406 QQueue<QQmlProfilerEvent> pointEvents;
414 const QQmlProfilerEvent &event,
const QQmlProfilerEventType &type)
416 QList<QQmlProfilerEvent> &starts = rangeStarts[type.rangeType()];
417 switch (event.rangeStage()) {
420 starts.append(event);
424 const qint64 invalidTimestamp = -1;
425 QList<qint64> &ends = rangeEnds[type.rangeType()];
428 ends.resize(starts.size(), invalidTimestamp);
430 qsizetype i = starts.size();
431 while (ends[--i] != invalidTimestamp) {}
434 Q_ASSERT(starts[i].timestamp() <= event.timestamp());
436 ends[i] = event.timestamp();
450 qsizetype index[MaximumRangeType] = { 0, 0, 0, 0, 0, 0 };
454 qsizetype minimum = MaximumRangeType;
455 qint64 minimumTime = std::numeric_limits<qint64>::max();
456 for (qsizetype i = 0; i < MaximumRangeType; ++i) {
457 const QList<QQmlProfilerEvent> &starts = rangeStarts[i];
458 if (starts.size() == index[i])
460 const qint64 timestamp = starts[index[i]].timestamp();
461 if (timestamp < minimumTime) {
462 minimumTime = timestamp;
466 if (minimum == MaximumRangeType)
470 while (!pointEvents.isEmpty() && pointEvents.front().timestamp() < minimumTime)
471 sendEvent(pointEvents.dequeue(), 0);
474 sendEvent(rangeStarts[minimum][index[minimum]],
475 rangeEnds[minimum][index[minimum]] - minimumTime);
485 for (qsizetype i = 0; i < MaximumRangeType; ++i) {
486 rangeStarts[i].clear();
487 rangeEnds[i].clear();
493 for (
const QQmlProfilerEvent &event : std::as_const(d->events)) {
494 const QQmlProfilerEventType &type = d->eventTypes.at(event.typeIndex());
495 if (type.rangeType() != MaximumRangeType)
496 handleRangeEvent(event, type);
500 pointEvents.enqueue(event);
503 for (qsizetype i = 0; i < MaximumRangeType; ++i) {
504 while (rangeEnds[i].size() < rangeStarts[i].size()) {
505 rangeEnds[i].append(d->traceEndTime);
515 Q_D(QQmlProfilerQtdWriter);
518 emit error(tr(
"No data to save"));
523 if (!stream.error.isEmpty()) {
524 emit error(stream.error);
529 stream.writeAttribute(
"traceStart", d->traceStartTime);
530 stream.writeAttribute(
"traceEnd", d->traceEndTime);
533 stream.writeAttribute(
"totalTime", d->qmlMeasuredTime);
535 for (
int typeIndex = 0, end = d->eventTypes.size(); typeIndex < end; ++typeIndex) {
536 const QQmlProfilerEventType &eventData = d->eventTypes.at(typeIndex);
538 stream.writeAttribute(
"index", typeIndex);
539 if (!eventData.displayName().isEmpty())
540 stream.writeTextElement(
"displayname", eventData.displayName());
542 stream.writeTextElement(
"type", eventData.rangeType() == MaximumRangeType
543 ? qmlMessageAsString(eventData.message())
544 : qmlRangeTypeAsString(eventData.rangeType()));
546 const QQmlProfilerEventLocation location = eventData.location();
547 if (!location.filename().isEmpty())
548 stream.writeTextElement(
"filename", location.filename());
549 if (location.line() >= 0)
550 stream.writeTextElement(
"line", location.line());
551 if (location.column() >= 0)
552 stream.writeTextElement(
"column", location.column());
553 if (!eventData.data().isEmpty())
554 stream.writeTextElement(
"details", eventData.data());
555 if (eventData.rangeType() == Binding)
556 stream.writeTextElement(
"bindingType", eventData.detailType());
557 else if (eventData.message() == Event) {
558 switch (eventData.detailType()) {
560 stream.writeTextElement(
"animationFrame", eventData.detailType());
563 stream.writeTextElement(
"keyEvent", eventData.detailType());
566 stream.writeTextElement(
"mouseEvent", eventData.detailType());
569 }
else if (eventData.message() == PixmapCacheEvent)
570 stream.writeTextElement(
"cacheEventType", eventData.detailType());
571 else if (eventData.message() == SceneGraphFrame)
572 stream.writeTextElement(
"sgEventType", eventData.detailType());
573 else if (eventData.message() == MemoryAllocation)
574 stream.writeTextElement(
"memoryEventType", eventData.detailType());
581 auto sendEvent = [&](
const QQmlProfilerEvent &event, qint64 duration = 0) {
582 Q_ASSERT(duration >= 0);
583 const QQmlProfilerEventType &type = d->eventTypes.at(event.typeIndex());
585 stream.writeAttribute(
"startTime", event.timestamp());
587 stream.writeAttribute(
"duration", duration);
588 stream.writeAttribute(
"eventIndex", event.typeIndex());
589 if (type.message() == Event) {
590 if (type.detailType() == AnimationFrame) {
592 stream.writeAttribute(
"framerate", event, 0);
593 stream.writeAttribute(
"animationcount", event, 1);
594 stream.writeAttribute(
"thread", event, 2);
595 }
else if (type.detailType() == Key || type.detailType() == Mouse) {
597 stream.writeAttribute(
"type", event, 0);
598 stream.writeAttribute(
"data1", event, 1);
599 stream.writeAttribute(
"data2", event, 2);
601 }
else if (type.message() == PixmapCacheEvent) {
603 if (type.detailType() == PixmapSizeKnown) {
604 stream.writeAttribute(
"width", event, 0);
605 stream.writeAttribute(
"height", event, 1);
606 }
else if (type.detailType() == PixmapReferenceCountChanged
607 || type.detailType() == PixmapCacheCountChanged) {
608 stream.writeAttribute(
"refCount", event, 1);
610 }
else if (type.message() == SceneGraphFrame) {
611 stream.writeAttribute(
"timing1", event, 0,
false);
612 stream.writeAttribute(
"timing2", event, 1,
false);
613 stream.writeAttribute(
"timing3", event, 2,
false);
614 stream.writeAttribute(
"timing4", event, 3,
false);
615 stream.writeAttribute(
"timing5", event, 4,
false);
616 }
else if (type.message() == MemoryAllocation) {
617 stream.writeAttribute(
"amount", event, 0);
622 DataIterator(d, std::move(sendEvent)).run();
632 if (state == newState)
635 Q_Q(QQmlProfilerQtdWriter);
640 emit q->error(
"Invalid qmlprofiler state change (Empty)"_L1);
644 if (state == ProcessingData)
645 emit q->error(
"Invalid qmlprofiler state change (AcquiringData)"_L1);
648 if (state != AcquiringData)
649 emit q->error(
"Invalid qmlprofiler state change (ProcessingData)"_L1);
652 if (state != ProcessingData && state != Empty)
653 emit q->error(
"Invalid qmlprofiler state change (Done)"_L1);
656 emit q->error(
"Trying to set unknown state in events list"_L1);
663 if (state == Done && isEmpty())
670 return d_func()->eventTypes.size();
675 return d_func()->events.size();
681#include "moc_qqmlprofilerqtdwriter_p.cpp"
QList< QQmlProfilerEventType > eventTypes
void setState(State state)
QList< QQmlProfilerEvent > events
void addEvent(const QQmlProfilerEvent &event) final
void complete(qint64 maximumTime) final
void endTrace(qint64 time, const QList< int > &engineIds) final
qsizetype numLoadedEventTypes() const final
qsizetype numLoadedEvents() const final
bool save(const QString &filename) final
void addEventType(const QQmlProfilerEventType &type) final
void startTrace(qint64 time, const QList< int > &engineIds) final
static const char * MESSAGE_STRINGS[]
static const char * RANGE_TYPE_STRINGS[]
static QString qmlRangeTypeAsString(RangeType type)
bool compareStartTimes(const QQmlProfilerEvent &t1, const QQmlProfilerEvent &t2)
const char PROFILER_FILE_VERSION[]
static QString qmlMessageAsString(Message type)
Q_STATIC_ASSERT(sizeof(SharedImageHeader) % 4==0)
DataIterator(const QQmlProfilerQtdWriterPrivate *d, qxp::function_ref< void(const QQmlProfilerEvent &, qint64)> &&sendEvent)
void writeStartElement(const char *name)
StreamWriter(const QString &filename)
void writeAttribute(const char *name, Number number)
void writeAttribute(const char *name, const char *value)
void writeTextElement(const char *name, Number number)
void writeTextElement(const char *name, const char *value)
void writeAttribute(const char *name, const QQmlProfilerEvent &event, int i, bool printZero=true)
void writeTextElement(const char *name, const QString &value)