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 *));
53 Q_DECLARE_PUBLIC(QQmlProfilerQtdWriter)
73 bool isEmpty()
const {
return events.isEmpty(); }
81 QQmlProfilerEventReceiver(*
new QQmlProfilerQtdWriterPrivate, parent)
89 Q_D(QQmlProfilerQtdWriter);
92 d->traceStartTime = std::numeric_limits<qint64>::max();
93 d->traceEndTime = std::numeric_limits<qint64>::min();
95 d->qmlMeasuredTime = 0;
96 d->setState(QQmlProfilerQtdWriterPrivate::Empty);
98 QQmlProfilerEventReceiver::clear();
103 QQmlProfilerEventReceiver::startTrace(time, engineIds);
105 Q_D(QQmlProfilerQtdWriter);
106 if (time < d->traceStartTime)
107 d->traceStartTime = time;
112 QQmlProfilerEventReceiver::endTrace(time, engineIds);
114 Q_D(QQmlProfilerQtdWriter);
115 if (time > d->traceEndTime)
116 d->traceEndTime = time;
125 return QString::number(type);
133 return QString::number(type);
138 Q_D(QQmlProfilerQtdWriter);
139 d->setState(QQmlProfilerQtdWriterPrivate::AcquiringData);
140 d->events.append(event);
145 QQmlProfilerEventType newType = type;
149 if (!type.data().isEmpty()) {
150 details = type.data().simplified();
151 QRegularExpression rewrite(QStringLiteral(
"^\\(function \\$(\\w+)\\(\\) \\{ (return |)(.+) \\}\\)$"));
152 QRegularExpressionMatch match = rewrite.match(details);
153 if (match.hasMatch()) {
154 details = match.captured(1) +QLatin1String(
": ") + match.captured(3);
156 if (details.startsWith(QLatin1String(
"file://")))
157 details = details.mid(details.lastIndexOf(QLatin1Char(
'/')) + 1);
160 newType.setData(details);
163 switch (type.message()) {
165 switch (type.detailType()) {
168 displayName = QString::fromLatin1(
"Input:%1").arg(type.detailType());
171 displayName = QString::fromLatin1(
"AnimationFrame");
174 displayName = QString::fromLatin1(
"Unknown");
185 case PixmapCacheEvent: {
186 const QString filePath = QUrl(type.location().filename()).path();
187 displayName = QStringView{filePath}.mid(filePath.lastIndexOf(QLatin1Char(
'/')) + 1)
188 + QLatin1Char(
':') + QString::number(type.detailType());
191 case SceneGraphFrame:
192 displayName = QString::fromLatin1(
"SceneGraph:%1").arg(type.detailType());
194 case MemoryAllocation:
195 displayName = QString::fromLatin1(
"MemoryAllocation:%1").arg(type.detailType());
198 displayName = QString::fromLatin1(
"DebugMessage:%1").arg(type.detailType());
200 case MaximumMessage: {
201 const QQmlProfilerEventLocation eventLocation = type.location();
203 if (eventLocation.filename().isEmpty()) {
204 displayName = QString::fromLatin1(
"Unknown");
206 const QString filePath = QUrl(eventLocation.filename()).path();
207 displayName = QStringView{filePath}.mid(
208 filePath.lastIndexOf(QLatin1Char(
'/')) + 1) +
209 QLatin1Char(
':') + QString::number(eventLocation.line());
215 newType.setDisplayName(displayName);
216 d_func()->eventTypes.append(newType);
222 qint64 level0Start = -1;
225 for (
const QQmlProfilerEvent &event : std::as_const(events)) {
226 const QQmlProfilerEventType &type = eventTypes.at(event.typeIndex());
227 if (type.message() != MaximumMessage)
230 switch (type.rangeType()) {
236 switch (event.rangeStage()) {
239 level0Start = event.timestamp();
243 qmlMeasuredTime += event.timestamp() - level0Start;
257 return t1.timestamp() < t2.timestamp();
262 if (events.size() < 2)
267 QVector<QQmlProfilerEvent>::iterator itFrom = events.end() - 2;
268 QVector<QQmlProfilerEvent>::iterator itTo = events.end() - 1;
270 while (itFrom != events.begin() && itTo != events.begin()) {
272 while (itFrom != events.begin() && itTo->timestamp() > itFrom->timestamp()) {
278 if (itFrom == events.begin())
282 while (itFrom != events.begin() && itTo->timestamp() <= itFrom->timestamp())
285 if (itTo->timestamp() <= itFrom->timestamp())
286 std::sort(itFrom, itTo + 1, compareStartTimes);
288 std::sort(itFrom + 1, itTo + 1, compareStartTimes);
298 Q_D(QQmlProfilerQtdWriter);
299 d->setState(QQmlProfilerQtdWriterPrivate::ProcessingData);
302 d->setState(QQmlProfilerQtdWriterPrivate::Done);
303 QQmlProfilerEventReceiver::complete(maximumTime);
308 return d_func()->isEmpty();
316 if (!filename.isEmpty()) {
317 file.setFileName(filename);
318 if (!file.open(QIODevice::WriteOnly)) {
319 error = QQmlProfilerQtdWriter::tr(
"Could not open %1 for writing").arg(filename);
323 if (!file.open(stdout, QIODevice::WriteOnly)) {
324 error = QQmlProfilerQtdWriter::tr(
"Could not open stdout for writing");
329 stream.setDevice(&file);
330 stream.setAutoFormatting(
true);
331 stream.writeStartDocument();
337 stream.writeEndDocument();
341 template<
typename Number>
344 stream.writeAttribute(QLatin1String(name), QString::number(number));
349 stream.writeAttribute(QLatin1String(name), QLatin1String(value));
352 void writeAttribute(
const char *name,
const QQmlProfilerEvent &event,
int i,
bool printZero =
true)
354 const qint64 number = event.number<qint64>(i);
355 if (printZero || number != 0)
356 writeAttribute(name, number);
359 template<
typename Number>
362 writeTextElement(name, QString::number(number));
367 stream.writeTextElement(QLatin1String(name), QLatin1String(value));
372 stream.writeTextElement(QLatin1String(name), value);
377 stream.writeStartElement(QLatin1String(name));
382 stream.writeEndElement();
387 QXmlStreamWriter stream;
394 qxp::function_ref<
void(
const QQmlProfilerEvent &, qint64)> &&sendEvent)
402 void handleRangeEvent(
const QQmlProfilerEvent &event,
const QQmlProfilerEventType &type);
407 const qxp::function_ref<
void(
const QQmlProfilerEvent &, qint64)> sendEvent;
409 QQueue<QQmlProfilerEvent> pointEvents;
417 const QQmlProfilerEvent &event,
const QQmlProfilerEventType &type)
419 QList<QQmlProfilerEvent> &starts = rangeStarts[type.rangeType()];
420 switch (event.rangeStage()) {
423 starts.append(event);
427 const qint64 invalidTimestamp = -1;
428 QList<qint64> &ends = rangeEnds[type.rangeType()];
431 ends.resize(starts.size(), invalidTimestamp);
433 qsizetype i = starts.size();
434 while (ends[--i] != invalidTimestamp) {}
437 Q_ASSERT(starts[i].timestamp() <= event.timestamp());
439 ends[i] = event.timestamp();
453 qsizetype index[MaximumRangeType] = { 0, 0, 0, 0, 0, 0 };
457 qsizetype minimum = MaximumRangeType;
458 qint64 minimumTime = std::numeric_limits<qint64>::max();
459 for (qsizetype i = 0; i < MaximumRangeType; ++i) {
460 const QList<QQmlProfilerEvent> &starts = rangeStarts[i];
461 if (starts.size() == index[i])
463 const qint64 timestamp = starts[index[i]].timestamp();
464 if (timestamp < minimumTime) {
465 minimumTime = timestamp;
469 if (minimum == MaximumRangeType)
473 while (!pointEvents.isEmpty() && pointEvents.front().timestamp() < minimumTime)
474 sendEvent(pointEvents.dequeue(), 0);
477 sendEvent(rangeStarts[minimum][index[minimum]],
478 rangeEnds[minimum][index[minimum]] - minimumTime);
488 for (qsizetype i = 0; i < MaximumRangeType; ++i) {
489 rangeStarts[i].clear();
490 rangeEnds[i].clear();
496 for (
const QQmlProfilerEvent &event : std::as_const(d->events)) {
497 const QQmlProfilerEventType &type = d->eventTypes.at(event.typeIndex());
498 if (type.rangeType() != MaximumRangeType)
499 handleRangeEvent(event, type);
503 pointEvents.enqueue(event);
506 for (qsizetype i = 0; i < MaximumRangeType; ++i) {
507 while (rangeEnds[i].size() < rangeStarts[i].size()) {
508 rangeEnds[i].append(d->traceEndTime);
518 Q_D(QQmlProfilerQtdWriter);
521 emit error(tr(
"No data to save"));
526 if (!stream.error.isEmpty()) {
527 emit error(stream.error);
532 stream.writeAttribute(
"traceStart", d->traceStartTime);
533 stream.writeAttribute(
"traceEnd", d->traceEndTime);
536 stream.writeAttribute(
"totalTime", d->qmlMeasuredTime);
538 for (
int typeIndex = 0, end = d->eventTypes.size(); typeIndex < end; ++typeIndex) {
539 const QQmlProfilerEventType &eventData = d->eventTypes.at(typeIndex);
541 stream.writeAttribute(
"index", typeIndex);
542 if (!eventData.displayName().isEmpty())
543 stream.writeTextElement(
"displayname", eventData.displayName());
545 stream.writeTextElement(
"type", eventData.rangeType() == MaximumRangeType
546 ? qmlMessageAsString(eventData.message())
547 : qmlRangeTypeAsString(eventData.rangeType()));
549 const QQmlProfilerEventLocation location = eventData.location();
550 if (!location.filename().isEmpty())
551 stream.writeTextElement(
"filename", location.filename());
552 if (location.line() >= 0)
553 stream.writeTextElement(
"line", location.line());
554 if (location.column() >= 0)
555 stream.writeTextElement(
"column", location.column());
556 if (!eventData.data().isEmpty())
557 stream.writeTextElement(
"details", eventData.data());
558 if (eventData.rangeType() == Binding)
559 stream.writeTextElement(
"bindingType", eventData.detailType());
560 else if (eventData.message() == Event) {
561 switch (eventData.detailType()) {
563 stream.writeTextElement(
"animationFrame", eventData.detailType());
566 stream.writeTextElement(
"keyEvent", eventData.detailType());
569 stream.writeTextElement(
"mouseEvent", eventData.detailType());
572 }
else if (eventData.message() == PixmapCacheEvent)
573 stream.writeTextElement(
"cacheEventType", eventData.detailType());
574 else if (eventData.message() == SceneGraphFrame)
575 stream.writeTextElement(
"sgEventType", eventData.detailType());
576 else if (eventData.message() == MemoryAllocation)
577 stream.writeTextElement(
"memoryEventType", eventData.detailType());
584 auto sendEvent = [&](
const QQmlProfilerEvent &event, qint64 duration = 0) {
585 Q_ASSERT(duration >= 0);
586 const QQmlProfilerEventType &type = d->eventTypes.at(event.typeIndex());
588 stream.writeAttribute(
"startTime", event.timestamp());
590 stream.writeAttribute(
"duration", duration);
591 stream.writeAttribute(
"eventIndex", event.typeIndex());
592 if (type.message() == Event) {
593 if (type.detailType() == AnimationFrame) {
595 stream.writeAttribute(
"framerate", event, 0);
596 stream.writeAttribute(
"animationcount", event, 1);
597 stream.writeAttribute(
"thread", event, 2);
598 }
else if (type.detailType() == Key || type.detailType() == Mouse) {
600 stream.writeAttribute(
"type", event, 0);
601 stream.writeAttribute(
"data1", event, 1);
602 stream.writeAttribute(
"data2", event, 2);
604 }
else if (type.message() == PixmapCacheEvent) {
606 if (type.detailType() == PixmapSizeKnown) {
607 stream.writeAttribute(
"width", event, 0);
608 stream.writeAttribute(
"height", event, 1);
609 }
else if (type.detailType() == PixmapReferenceCountChanged
610 || type.detailType() == PixmapCacheCountChanged) {
611 stream.writeAttribute(
"refCount", event, 1);
613 }
else if (type.message() == SceneGraphFrame) {
614 stream.writeAttribute(
"timing1", event, 0,
false);
615 stream.writeAttribute(
"timing2", event, 1,
false);
616 stream.writeAttribute(
"timing3", event, 2,
false);
617 stream.writeAttribute(
"timing4", event, 3,
false);
618 stream.writeAttribute(
"timing5", event, 4,
false);
619 }
else if (type.message() == MemoryAllocation) {
620 stream.writeAttribute(
"amount", event, 0);
625 DataIterator(d, std::move(sendEvent)).run();
635 if (state == newState)
638 Q_Q(QQmlProfilerQtdWriter);
643 emit q->error(
"Invalid qmlprofiler state change (Empty)"_L1);
647 if (state == ProcessingData)
648 emit q->error(
"Invalid qmlprofiler state change (AcquiringData)"_L1);
651 if (state != AcquiringData)
652 emit q->error(
"Invalid qmlprofiler state change (ProcessingData)"_L1);
655 if (state != ProcessingData && state != Empty)
656 emit q->error(
"Invalid qmlprofiler state change (Done)"_L1);
659 emit q->error(
"Trying to set unknown state in events list"_L1);
666 if (state == Done && isEmpty())
673 return d_func()->eventTypes.size();
678#include "moc_qqmlprofilerqtdwriter_p.cpp"
QVector< QQmlProfilerEvent > events
void setState(State state)
QVector< QQmlProfilerEventType > eventTypes
void addEvent(const QQmlProfilerEvent &event) final
void complete(qint64 maximumTime) final
void endTrace(qint64 time, const QList< int > &engineIds) final
bool save(const QString &filename) final
void addEventType(const QQmlProfilerEventType &type) final
bool isEmpty() const final
int numLoadedEventTypes() const 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)