/**************************************************************************** ** ** Copyright (C) 2013 Digia Plc and/or its subsidiary(-ies). ** Contact: http://www.qt-project.org/legal ** ** This file is part of the QtQml module of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL$ ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and Digia. For licensing terms and ** conditions see http://qt.digia.com/licensing. For further information ** use the contact form at http://qt.digia.com/contact-us. ** ** GNU Lesser General Public License Usage ** Alternatively, this file may be used under the terms of the GNU Lesser ** General Public License version 2.1 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 2.1 requirements ** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. ** ** In addition, as a special exception, Digia gives you certain additional ** rights. These rights are described in the Digia Qt LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3.0 as published by the Free Software ** Foundation and appearing in the file LICENSE.GPL included in the ** packaging of this file. Please review the following information to ** ensure the GNU General Public License version 3.0 requirements will be ** met: http://www.gnu.org/copyleft/gpl.html. ** ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #ifndef QQMLPROFILERSERVICE_P_H #define QQMLPROFILERSERVICE_P_H // // W A R N I N G // ------------- // // This file is not part of the Qt API. It exists purely as an // implementation detail. This header file may change from version to // version without notice, or even be removed. // // We mean it. // #include #include #include #include #include #include #include #include QT_BEGIN_NAMESPACE struct Q_AUTOTEST_EXPORT QQmlProfilerData { qint64 time; int messageType; int detailType; //### QString detailData; //used by RangeData and RangeLocation int line; //used by RangeLocation, also as "width" for pixmaps int column; //used by RangeLocation, also as "height" for pixmaps int framerate; //used by animation events int animationcount; //used by animation events, also as "cache/reference count" for pixmaps int bindingType; qint64 subtime_1; qint64 subtime_2; qint64 subtime_3; qint64 subtime_4; qint64 subtime_5; QByteArray toByteArray() const; }; Q_DECLARE_TYPEINFO(QQmlProfilerData, Q_MOVABLE_TYPE); class QUrl; class QQmlEngine; class Q_QML_PRIVATE_EXPORT QQmlProfilerService : public QQmlDebugService { public: enum Message { Event, RangeStart, RangeData, RangeLocation, RangeEnd, Complete, // end of transmission PixmapCacheEvent, SceneGraphFrame, MaximumMessage }; enum EventType { FramePaint, Mouse, Key, AnimationFrame, EndTrace, StartTrace, MaximumEventType }; enum RangeType { Painting, Compiling, Creating, Binding, //running a binding HandlingSignal, //running a signal handler MaximumRangeType }; enum BindingType { QmlBinding, V8Binding, V4Binding, MaximumBindingType }; enum PixmapEventType { PixmapSizeKnown, PixmapReferenceCountChanged, PixmapCacheCountChanged, PixmapLoadingStarted, PixmapLoadingFinished, PixmapLoadingError, MaximumPixmapEventType }; enum SceneGraphFrameType { SceneGraphRendererFrame, SceneGraphAdaptationLayerFrame, SceneGraphContextFrame, SceneGraphRenderLoopFrame, SceneGraphTexturePrepare, SceneGraphTextureDeletion, SceneGraphPolishAndSync, SceneGraphWindowsRenderShow, SceneGraphWindowsAnimations, SceneGraphWindowsPolishFrame, MaximumSceneGraphFrameType }; static void initialize(); static bool startProfiling(); static bool stopProfiling(); static void sendStartedProfilingMessage(); static bool profilingEnabled(); static void addEvent(EventType); static void animationFrame(qint64); static void sceneGraphFrame(SceneGraphFrameType frameType, qint64 value1, qint64 value2 = -1, qint64 value3 = -1, qint64 value4 = -1, qint64 value5 = -1); static void sendProfilingData(); QQmlProfilerService(); ~QQmlProfilerService(); protected: virtual void stateAboutToBeChanged(State state); virtual void messageReceived(const QByteArray &); private: bool startProfilingImpl(); bool stopProfilingImpl(); void sendStartedProfilingMessageImpl(); void addEventImpl(EventType); void animationFrameImpl(qint64); void startRange(RangeType, BindingType bindingType = QmlBinding); void rangeData(RangeType, const QString &); void rangeData(RangeType, const QUrl &); void rangeLocation(RangeType, const QString &, int, int); void rangeLocation(RangeType, const QUrl &, int, int); void endRange(RangeType); // overloading depending on parameters void pixmapEventImpl(PixmapEventType eventType, const QUrl &url); void pixmapEventImpl(PixmapEventType eventType, const QUrl &url, int width, int height); void pixmapEventImpl(PixmapEventType eventType, const QUrl &url, int count); void sceneGraphFrameImpl(SceneGraphFrameType frameType, qint64 value1, qint64 value2, qint64 value3, qint64 value4, qint64 value5); void setProfilingEnabled(bool enable); void sendMessages(); void processMessage(const QQmlProfilerData &); public: static bool enabled; private: QElapsedTimer m_timer; QVector m_data; QMutex m_dataMutex; QMutex m_initializeMutex; QWaitCondition m_initializeCondition; static QQmlProfilerService *instance; friend struct QQmlBindingProfiler; friend struct QQmlHandlingSignalProfiler; friend struct QQmlObjectCreatingProfiler; friend struct QQmlCompilingProfiler; friend struct QQmlPixmapProfiler; }; // // RAII helper structs // struct QQmlBindingProfiler { QQmlBindingProfiler(const QString &url, int line, int column, QQmlProfilerService::BindingType bindingType) { QQmlProfilerService *instance = QQmlProfilerService::instance; enabled = instance ? instance->profilingEnabled() : false; if (enabled) { instance->startRange(QQmlProfilerService::Binding, bindingType); instance->rangeLocation(QQmlProfilerService::Binding, url, line, column); } } ~QQmlBindingProfiler() { if (enabled) QQmlProfilerService::instance->endRange(QQmlProfilerService::Binding); } bool enabled; }; struct QQmlHandlingSignalProfiler { QQmlHandlingSignalProfiler(QQmlBoundSignalExpression *expression) { enabled = QQmlProfilerService::instance ? QQmlProfilerService::instance->profilingEnabled() : false; if (enabled) { QQmlProfilerService *service = QQmlProfilerService::instance; service->startRange(QQmlProfilerService::HandlingSignal); service->rangeLocation(QQmlProfilerService::HandlingSignal, expression->sourceFile(), expression->lineNumber(), expression->columnNumber()); } } ~QQmlHandlingSignalProfiler() { if (enabled) QQmlProfilerService::instance->endRange(QQmlProfilerService::HandlingSignal); } bool enabled; }; struct QQmlObjectCreatingProfiler { QQmlObjectCreatingProfiler() { enabled = QQmlProfilerService::instance ? QQmlProfilerService::instance->profilingEnabled() : false; if (enabled) { QQmlProfilerService *service = QQmlProfilerService::instance; service->startRange(QQmlProfilerService::Creating); } } void setTypeName(const QString &typeName) { Q_ASSERT_X(enabled, Q_FUNC_INFO, "method called although profiler is not enabled."); QQmlProfilerService::instance->rangeData(QQmlProfilerService::Creating, typeName); } void setLocation(const QUrl &url, int line, int column) { Q_ASSERT_X(enabled, Q_FUNC_INFO, "method called although profiler is not enabled."); if (enabled) QQmlProfilerService::instance->rangeLocation( QQmlProfilerService::Creating, url, line, column); } ~QQmlObjectCreatingProfiler() { if (enabled) QQmlProfilerService::instance->endRange(QQmlProfilerService::Creating); } bool enabled; }; struct QQmlCompilingProfiler { QQmlCompilingProfiler(const QString &name) { QQmlProfilerService *instance = QQmlProfilerService::instance; enabled = instance ? instance->profilingEnabled() : false; if (enabled) { instance->startRange(QQmlProfilerService::Compiling); instance->rangeLocation(QQmlProfilerService::Compiling, name, 1, 1); instance->rangeData(QQmlProfilerService::Compiling, name); } } ~QQmlCompilingProfiler() { if (enabled) QQmlProfilerService::instance->endRange(QQmlProfilerService::Compiling); } bool enabled; }; struct QQmlPixmapProfiler { QQmlPixmapProfiler() { QQmlProfilerService *instance = QQmlProfilerService::instance; enabled = instance ? instance->profilingEnabled() : false; } ~QQmlPixmapProfiler() {} void startLoading(const QUrl &pixmapUrl) { if (enabled) { QQmlProfilerService::instance->pixmapEventImpl(QQmlProfilerService::PixmapLoadingStarted, pixmapUrl); } } void finishLoading(const QUrl &pixmapUrl) { if (enabled) { QQmlProfilerService::instance->pixmapEventImpl(QQmlProfilerService::PixmapLoadingFinished, pixmapUrl); } } void errorLoading(const QUrl &pixmapUrl) { if (enabled) { QQmlProfilerService::instance->pixmapEventImpl(QQmlProfilerService::PixmapLoadingError, pixmapUrl); } } void cacheCountChanged(const QUrl &pixmapUrl, int cacheCount) { if (enabled) { QQmlProfilerService::instance->pixmapEventImpl(QQmlProfilerService::PixmapCacheCountChanged, pixmapUrl, cacheCount); } } void referenceCountChanged(const QUrl &pixmapUrl, int referenceCount) { if (enabled) { QQmlProfilerService::instance->pixmapEventImpl(QQmlProfilerService::PixmapReferenceCountChanged, pixmapUrl, referenceCount); } } void setSize(const QUrl &pixmapUrl, int width, int height) { if (enabled) { QQmlProfilerService::instance->pixmapEventImpl(QQmlProfilerService::PixmapSizeKnown, pixmapUrl, width, height); } } bool enabled; }; QT_END_NAMESPACE #endif // QQMLPROFILERSERVICE_P_H